* [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
@ 2023-08-25 15:24 Qing Zhao
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
` (6 more replies)
0 siblings, 7 replies; 116+ messages in thread
From: Qing Zhao @ 2023-08-25 15:24 UTC (permalink / raw)
To: joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard, Qing Zhao
This is the 3rd version of the patch, per our discussion based on the
review comments for the 1st and 2nd version, the major changes in this
version are:
***Against 1st version:
1. change the name "element_count" to "counted_by";
2. change the parameter for the attribute from a STRING to an
Identifier;
3. Add logic and testing cases to handle anonymous structure/unions;
4. Clarify documentation to permit the situation when the allocation
size is larger than what's specified by "counted_by", at the same time,
it's user's error if allocation size is smaller than what's specified by
"counted_by";
5. Add a complete testing case for using counted_by attribute in
__builtin_dynamic_object_size when there is mismatch between the
allocation size and the value of "counted_by", the expecting behavior
for each case and the explanation on why in the comments.
***Against 2rd version:
1. Identify a tree node sharing issue and fixed it in the routine
"component_ref_get_counted_ty" of tree.cc;
2. Update the documentation and testing cases with the clear usage
of the fomula to compute the allocation size:
MAX (sizeof (struct A), offsetof (struct A, array[0]) + counted_by * sizeof(element))
(the algorithm used in tree-object-size.cc is correct).
In this set of patches, the major functionality provided is:
1. a new attribute "counted_by";
2. use this new attribute in bound sanitizer;
3. use this new attribute in dynamic object size for subobject size;
As discussed, I plan to add two more separate patches sets after this initial
patch set is approved and committed.
set 1. A new warning option and a new sanitizer option for the user error
when the allocation size is smaller than the value of "counted_by".
set 2. An improvement to __builtin_dynamic_object_size for whole-object
size of the structure with FAM annaoted with counted_by.
there are also some existing bugs in tree-object-size.cc identified
during the study, and PRs were filed to record them. these bugs will
be fixed seperately with individual patches:
https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111030
https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111040
Bootstrapped and regression tested on both aarch64 and X86, no issue.
Please see more details on the description of this work on:
https://gcc.gnu.org/pipermail/gcc-patches/2023-May/619708.html
and more discussions on
https://gcc.gnu.org/pipermail/gcc-patches/2023-August/626376.html
Okay for committing?
thanks.
Qing
Qing Zhao (3):
Provide counted_by attribute to flexible array member field (PR108896)
Use the counted_by atribute info in builtin object size [PR108896]
Use the counted_by attribute information in bound sanitizer[PR108896]
gcc/c-family/c-attribs.cc | 54 ++++-
gcc/c-family/c-common.cc | 13 ++
gcc/c-family/c-common.h | 1 +
gcc/c-family/c-ubsan.cc | 16 ++
gcc/c/c-decl.cc | 79 +++++--
gcc/doc/extend.texi | 77 +++++++
.../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
.../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++
.../ubsan/flex-array-counted-by-bounds-2.c | 27 +++
.../ubsan/flex-array-counted-by-bounds.c | 46 ++++
gcc/tree-object-size.cc | 37 ++-
gcc/tree.cc | 133 +++++++++++
gcc/tree.h | 15 ++
14 files changed, 797 insertions(+), 25 deletions(-)
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
--
2.31.1
^ permalink raw reply [flat|nested] 116+ messages in thread
* [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
@ 2023-08-25 15:24 ` Qing Zhao
2023-09-08 14:12 ` Qing Zhao
` (2 more replies)
2023-08-25 15:24 ` [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896] Qing Zhao
` (5 subsequent siblings)
6 siblings, 3 replies; 116+ messages in thread
From: Qing Zhao @ 2023-08-25 15:24 UTC (permalink / raw)
To: joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard, Qing Zhao
Provide a new counted_by attribute to flexible array member field.
'counted_by (COUNT)'
The 'counted_by' attribute may be attached to the flexible array
member of a structure. It indicates that the number of the
elements of the array is given by the field named "COUNT" in the
same structure as the flexible array member. GCC uses this
information to improve the results of the array bound sanitizer and
the '__builtin_dynamic_object_size'.
For instance, the following code:
struct P {
size_t count;
char other;
char array[] __attribute__ ((counted_by (count)));
} *p;
specifies that the 'array' is a flexible array member whose number
of elements is given by the field 'count' in the same structure.
The field that represents the number of the elements should have an
integer type. An explicit 'counted_by' annotation defines a
relationship between two objects, 'p->array' and 'p->count', that
'p->array' has _at least_ 'p->count' number of elements available.
This relationship must hold even after any of these related objects
are updated. It's the user's responsibility to make sure this
relationship to be kept all the time. Otherwise the results of the
array bound sanitizer and the '__builtin_dynamic_object_size' might
be incorrect.
For instance, in the following example, the allocated array has
less elements than what's specified by the 'sbuf->count', this is
an user error. As a result, out-of-bounds access to the array
might not be detected.
#define SIZE_BUMP 10
struct P *sbuf;
void alloc_buf (size_t nelems)
{
sbuf = (struct P *) malloc (MAX (sizeof (struct P),
(offsetof (struct P, array[0])
+ nelems * sizeof (char))));
sbuf->count = nelems + SIZE_BUMP;
/* This is invalid when the sbuf->array has less than sbuf->count
elements. */
}
In the following example, the 2nd update to the field 'sbuf->count'
of the above structure will permit out-of-bounds access to the
array 'sbuf>array' as well.
#define SIZE_BUMP 10
struct P *sbuf;
void alloc_buf (size_t nelems)
{
sbuf = (struct P *) malloc (MAX (sizeof (struct P),
(offsetof (struct P, array[0])
+ (nelems + SIZE_BUMP) * sizeof (char))));
sbuf->count = nelems;
/* This is valid when the sbuf->array has at least sbuf->count
elements. */
}
void use_buf (int index)
{
sbuf->count = sbuf->count + SIZE_BUMP + 1;
/* Now the value of sbuf->count is larger than the number
of elements of sbuf->array. */
sbuf->array[index] = 0;
/* then the out-of-bound access to this array
might not be detected. */
}
gcc/c-family/ChangeLog:
PR C/108896
* c-attribs.cc (handle_counted_by_attribute): New function.
(attribute_takes_identifier_p): Add counted_by attribute to the list.
* c-common.cc (c_flexible_array_member_type_p): ...To this.
* c-common.h (c_flexible_array_member_type_p): New prototype.
gcc/c/ChangeLog:
PR C/108896
* c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
(add_flexible_array_elts_to_size): Use renamed function.
(is_flexible_array_member_p): Use renamed function.
(verify_counted_by_attribute): New function.
(finish_struct): Use renamed function and verify counted_by
attribute.
gcc/ChangeLog:
PR C/108896
* doc/extend.texi: Document attribute counted_by.
* tree.cc (get_named_field): New function.
* tree.h (get_named_field): New prototype.
gcc/testsuite/ChangeLog:
PR C/108896
* gcc.dg/flex-array-counted-by.c: New test.
---
gcc/c-family/c-attribs.cc | 54 ++++++++++++-
gcc/c-family/c-common.cc | 13 ++++
gcc/c-family/c-common.h | 1 +
gcc/c/c-decl.cc | 79 +++++++++++++++-----
gcc/doc/extend.texi | 77 +++++++++++++++++++
gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
gcc/tree.cc | 40 ++++++++++
gcc/tree.h | 5 ++
8 files changed, 291 insertions(+), 18 deletions(-)
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
index e2792ca6898b..65e4f6639109 100644
--- a/gcc/c-family/c-attribs.cc
+++ b/gcc/c-family/c-attribs.cc
@@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute (tree *, tree, tree,
int, bool *);
static tree handle_strict_flex_array_attribute (tree *, tree, tree,
int, bool *);
+static tree handle_counted_by_attribute (tree *, tree, tree,
+ int, bool *);
static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree, bool *);
@@ -373,6 +375,8 @@ const struct attribute_spec c_common_attribute_table[] =
handle_warn_if_not_aligned_attribute, NULL },
{ "strict_flex_array", 1, 1, true, false, false, false,
handle_strict_flex_array_attribute, NULL },
+ { "counted_by", 1, 1, true, false, false, false,
+ handle_counted_by_attribute, NULL },
{ "weak", 0, 0, true, false, false, false,
handle_weak_attribute, NULL },
{ "noplt", 0, 0, true, false, false, false,
@@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
else if (!strcmp ("mode", spec->name)
|| !strcmp ("format", spec->name)
|| !strcmp ("cleanup", spec->name)
- || !strcmp ("access", spec->name))
+ || !strcmp ("access", spec->name)
+ || !strcmp ("counted_by", spec->name))
return true;
else
return targetm.attribute_takes_identifier_p (attr_id);
@@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node, tree name,
return NULL_TREE;
}
+/* Handle a "counted_by" attribute; arguments as in
+ struct attribute_spec.handler. */
+
+static tree
+handle_counted_by_attribute (tree *node, tree name,
+ tree args, int ARG_UNUSED (flags),
+ bool *no_add_attrs)
+{
+ tree decl = *node;
+ tree argval = TREE_VALUE (args);
+
+ /* This attribute only applies to field decls of a structure. */
+ if (TREE_CODE (decl) != FIELD_DECL)
+ {
+ error_at (DECL_SOURCE_LOCATION (decl),
+ "%qE attribute may not be specified for non-field"
+ " declaration %q+D", name, decl);
+ *no_add_attrs = true;
+ }
+ /* This attribute only applies to field with array type. */
+ else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
+ {
+ error_at (DECL_SOURCE_LOCATION (decl),
+ "%qE attribute may not be specified for a non-array field",
+ name);
+ *no_add_attrs = true;
+ }
+ /* This attribute only applies to a C99 flexible array member type. */
+ else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
+ {
+ error_at (DECL_SOURCE_LOCATION (decl),
+ "%qE attribute may not be specified for a non"
+ " flexible array member field",
+ name);
+ *no_add_attrs = true;
+ }
+ /* The argument should be an identifier. */
+ else if (TREE_CODE (argval) != IDENTIFIER_NODE)
+ {
+ error_at (DECL_SOURCE_LOCATION (decl),
+ "%<counted_by%> argument not an identifier");
+ *no_add_attrs = true;
+ }
+
+ return NULL_TREE;
+}
+
/* Handle a "weak" attribute; arguments as in
struct attribute_spec.handler. */
diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
index 9fbaeb437a12..a18937245c2a 100644
--- a/gcc/c-family/c-common.cc
+++ b/gcc/c-family/c-common.cc
@@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
(*debug_hooks->early_global_decl) (cnode->decl);
}
+/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
+bool
+c_flexible_array_member_type_p (const_tree type)
+{
+ if (TREE_CODE (type) == ARRAY_TYPE
+ && TYPE_SIZE (type) == NULL_TREE
+ && TYPE_DOMAIN (type) != NULL_TREE
+ && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
+ return true;
+
+ return false;
+}
+
/* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based on the
values of attribute strict_flex_array and the flag_strict_flex_arrays. */
unsigned int
diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
index 78fc5248ba68..c29bb429062b 100644
--- a/gcc/c-family/c-common.h
+++ b/gcc/c-family/c-common.h
@@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
extern tree c_common_get_narrower (tree, int *);
extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
extern void c_common_finalize_early_debug (void);
+extern bool c_flexible_array_member_type_p (const_tree);
extern unsigned int c_strict_flex_array_level_of (tree);
extern bool c_option_is_from_cpp_diagnostics (int);
diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
index 1f9eb44dbaa2..e943b49b5230 100644
--- a/gcc/c/c-decl.cc
+++ b/gcc/c/c-decl.cc
@@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator *decl,
return decl;
}
-/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
-static bool
-flexible_array_member_type_p (const_tree type)
-{
- if (TREE_CODE (type) == ARRAY_TYPE
- && TYPE_SIZE (type) == NULL_TREE
- && TYPE_DOMAIN (type) != NULL_TREE
- && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
- return true;
-
- return false;
-}
-
/* Determine whether TYPE is a one-element array type "[1]". */
static bool
one_element_array_type_p (const_tree type)
@@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree init)
elt = CONSTRUCTOR_ELTS (init)->last ().value;
type = TREE_TYPE (elt);
- if (flexible_array_member_type_p (type))
+ if (c_flexible_array_member_type_p (type))
{
complete_array_type (&type, elt, false);
DECL_SIZE (decl)
@@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
- bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
+ bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE (x));
unsigned int strict_flex_array_level = c_strict_flex_array_level_of (x);
@@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
return false;
}
+/* Verify the argument of the counted_by attribute of the flexible array
+ member FIELD_DECL is a valid field of the containing structure's fieldlist,
+ FIELDLIST, Report error and remove this attribute when it's not. */
+static void
+verify_counted_by_attribute (tree fieldlist, tree field_decl)
+{
+ tree attr_counted_by = lookup_attribute ("counted_by",
+ DECL_ATTRIBUTES (field_decl));
+
+ if (!attr_counted_by)
+ return;
+
+ /* If there is an counted_by attribute attached to the field,
+ verify it. */
+
+ const char *fieldname
+ = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
+
+ /* Verify the argument of the attrbute is a valid field of the
+ containing structure. */
+
+ tree counted_by_field = get_named_field (fieldlist, fieldname);
+
+ /* Error when the field is not found in the containing structure. */
+ if (!counted_by_field)
+ {
+ error_at (DECL_SOURCE_LOCATION (field_decl),
+ "%qE attribute argument not a field declaration"
+ " in the same structure, ignore it",
+ (get_attribute_name (attr_counted_by)));
+
+ DECL_ATTRIBUTES (field_decl)
+ = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
+ }
+ else
+ /* Error when the field is not with an integer type. */
+ {
+ while (TREE_CHAIN (counted_by_field))
+ counted_by_field = TREE_CHAIN (counted_by_field);
+ tree real_field = TREE_VALUE (counted_by_field);
+
+ if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
+ {
+ error_at (DECL_SOURCE_LOCATION (field_decl),
+ "%qE attribute argument not a field declaration"
+ " with integer type, ignore it",
+ (get_attribute_name (attr_counted_by)));
+
+ DECL_ATTRIBUTES (field_decl)
+ = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
+ }
+ }
+
+ return;
+}
/* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
@@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
DECL_PACKED (x) = 1;
/* Detect flexible array member in an invalid context. */
- if (flexible_array_member_type_p (TREE_TYPE (x)))
+ if (c_flexible_array_member_type_p (TREE_TYPE (x)))
{
if (TREE_CODE (t) == UNION_TYPE)
{
@@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
"members");
TREE_TYPE (x) = error_mark_node;
}
+ /* if there is a counted_by attribute attached to this field,
+ verify it. */
+ verify_counted_by_attribute (fieldlist, x);
}
if (pedantic && TREE_CODE (t) == RECORD_TYPE
@@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
when x is an array and is the last field. */
if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
TYPE_INCLUDES_FLEXARRAY (t)
- = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
+ = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
/* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of x, t
when x is an union or record and is the last field. */
else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
index 97eaacf8a7ec..ea6240646936 100644
--- a/gcc/doc/extend.texi
+++ b/gcc/doc/extend.texi
@@ -7617,6 +7617,83 @@ When both the attribute and the option present at the same time, the level of
the strictness for the specific trailing array field is determined by the
attribute.
+@cindex @code{counted_by} variable attribute
+@item counted_by (@var{count})
+The @code{counted_by} attribute may be attached to the flexible array
+member of a structure. It indicates that the number of the elements of the
+array is given by the field named "@var{count}" in the same structure as the
+flexible array member. GCC uses this information to improve the results of
+the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
+
+For instance, the following code:
+
+@smallexample
+struct P @{
+ size_t count;
+ char other;
+ char array[] __attribute__ ((counted_by (count)));
+@} *p;
+@end smallexample
+
+@noindent
+specifies that the @code{array} is a flexible array member whose number of
+elements is given by the field @code{count} in the same structure.
+
+The field that represents the number of the elements should have an integer
+type. An explicit @code{counted_by} annotation defines a relationship between
+two objects, @code{p->array} and @code{p->count}, that @code{p->array} has
+@emph{at least} @code{p->count} number of elements available. This relationship
+must hold even after any of these related objects are updated. It's the user's
+responsibility to make sure this relationship to be kept all the time.
+Otherwise the results of the array bound sanitizer and the
+@code{__builtin_dynamic_object_size} might be incorrect.
+
+For instance, in the following example, the allocated array has less elements
+than what's specified by the @code{sbuf->count}, this is an user error. As a
+result, out-of-bounds access to the array might not be detected.
+
+@smallexample
+#define SIZE_BUMP 10
+struct P *sbuf;
+void alloc_buf (size_t nelems)
+@{
+ sbuf = (struct P *) malloc (MAX (sizeof (struct P),
+ (offsetof (struct P, array[0])
+ + nelems * sizeof (char))));
+ sbuf->count = nelems + SIZE_BUMP;
+ /* This is invalid when the sbuf->array has less than sbuf->count
+ elements. */
+@}
+@end smallexample
+
+In the following example, the 2nd update to the field @code{sbuf->count} of
+the above structure will permit out-of-bounds access to the array
+@code{sbuf>array} as well.
+
+@smallexample
+#define SIZE_BUMP 10
+struct P *sbuf;
+void alloc_buf (size_t nelems)
+@{
+ sbuf = (struct P *) malloc (MAX (sizeof (struct P),
+ (offsetof (struct P, array[0])
+ + (nelems + SIZE_BUMP) * sizeof (char))));
+ sbuf->count = nelems;
+ /* This is valid when the sbuf->array has at least sbuf->count
+ elements. */
+@}
+void use_buf (int index)
+@{
+ sbuf->count = sbuf->count + SIZE_BUMP + 1;
+ /* Now the value of sbuf->count is larger than the number
+ of elements of sbuf->array. */
+ sbuf->array[index] = 0;
+ /* then the out-of-bound access to this array
+ might not be detected. */
+@}
+@end smallexample
+
+
@cindex @code{alloc_size} variable attribute
@item alloc_size (@var{position})
@itemx alloc_size (@var{position-1}, @var{position-2})
diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
new file mode 100644
index 000000000000..f8ce9776bf86
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
@@ -0,0 +1,40 @@
+/* testing the correct usage of attribute counted_by. */
+/* { dg-do compile } */
+/* { dg-options "-O2" } */
+
+#include <wchar.h>
+
+int size;
+int x __attribute ((counted_by (size))); /* { dg-error "attribute may not be specified for non-field declaration" } */
+
+struct trailing {
+ int count;
+ int field __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non-array field" } */
+};
+
+struct trailing_1 {
+ int count;
+ int array_1[0] __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non flexible array member field" } */
+};
+
+int count;
+struct trailing_array_2 {
+ int count;
+ int array_2[] __attribute ((counted_by ("count"))); /* { dg-error "argument not an identifier" } */
+};
+
+struct trailing_array_3 {
+ int other;
+ int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error "argument not an identifier" } */
+};
+
+struct trailing_array_4 {
+ int other;
+ int array_4[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration in the same structure, ignore it" } */
+};
+
+int count;
+struct trailing_array_5 {
+ float count;
+ int array_5[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration with integer type, ignore it" } */
+};
diff --git a/gcc/tree.cc b/gcc/tree.cc
index 420857b110c4..fcd36ae0cd74 100644
--- a/gcc/tree.cc
+++ b/gcc/tree.cc
@@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
}
+/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
+ with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
+ whose name is FIELDNAME, which is the last TREE_VALUE of the list.
+ return NULL_TREE if such field is not found. Normally this list is of
+ length one, but if the field is embedded with (nested) anonymous structures
+ or unions, this list steps down the chain to the field. */
+tree
+get_named_field (tree fieldlist, const char *fieldname)
+{
+ tree named_field = NULL_TREE;
+ for (tree field = fieldlist; field; field = DECL_CHAIN (field))
+ {
+ if (TREE_CODE (field) != FIELD_DECL)
+ continue;
+ if (DECL_NAME (field) != NULL)
+ if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
+ {
+ named_field = tree_cons (NULL_TREE, field, named_field);
+ break;
+ }
+ else
+ continue;
+ /* if the field is an anonymous struct/union, we will check the nested
+ fields inside it recursively. */
+ else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
+ if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
+ fieldname)) != NULL_TREE)
+ {
+ named_field = tree_cons (NULL_TREE, field, named_field);
+ break;
+ }
+ else
+ continue;
+ else
+ continue;
+ }
+ return named_field;
+}
+
+
/* Return a tree representing the lower bound of the array mentioned in
EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
diff --git a/gcc/tree.h b/gcc/tree.h
index 4c04245e2b1b..4859becaa1e7 100644
--- a/gcc/tree.h
+++ b/gcc/tree.h
@@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
extern tree array_ref_element_size (tree);
+/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
+ name is FIELDNAME, return NULL_TREE if such field is not found.
+ searching nested anonymous structure/union recursively. */
+extern tree get_named_field (tree, const char *);
+
/* Return a typenode for the "standard" C type with a given name. */
extern tree get_typenode_from_name (const char *);
--
2.31.1
^ permalink raw reply [flat|nested] 116+ messages in thread
* [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896]
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
@ 2023-08-25 15:24 ` Qing Zhao
2023-09-08 14:12 ` Qing Zhao
` (2 more replies)
2023-08-25 15:24 ` [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896] Qing Zhao
` (4 subsequent siblings)
6 siblings, 3 replies; 116+ messages in thread
From: Qing Zhao @ 2023-08-25 15:24 UTC (permalink / raw)
To: joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard, Qing Zhao
Use the counted_by atribute info in builtin object size to compute the
subobject size for flexible array members.
gcc/ChangeLog:
PR C/108896
* tree-object-size.cc (addr_object_size): Use the counted_by
attribute info.
* tree.cc (component_ref_has_counted_by_p): New function.
(component_ref_get_counted_by): New function.
* tree.h (component_ref_has_counted_by_p): New prototype.
(component_ref_get_counted_by): New prototype.
gcc/testsuite/ChangeLog:
PR C/108896
* gcc.dg/flex-array-counted-by-2.c: New test.
* gcc.dg/flex-array-counted-by-3.c: New test.
---
.../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
.../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
gcc/tree-object-size.cc | 37 ++-
gcc/tree.cc | 95 +++++++-
gcc/tree.h | 10 +
5 files changed, 418 insertions(+), 8 deletions(-)
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
new file mode 100644
index 000000000000..ec580c1f1f01
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
@@ -0,0 +1,74 @@
+/* test the attribute counted_by and its usage in
+ * __builtin_dynamic_object_size. */
+/* { dg-do run } */
+/* { dg-options "-O2" } */
+
+#include "builtin-object-size-common.h"
+
+#define expect(p, _v) do { \
+ size_t v = _v; \
+ if (p == v) \
+ __builtin_printf ("ok: %s == %zd\n", #p, p); \
+ else \
+ { \
+ __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
+ FAIL (); \
+ } \
+} while (0);
+
+struct flex {
+ int b;
+ int c[];
+} *array_flex;
+
+struct annotated {
+ int b;
+ int c[] __attribute__ ((counted_by (b)));
+} *array_annotated;
+
+struct nested_annotated {
+ struct {
+ union {
+ int b;
+ float f;
+ };
+ int n;
+ };
+ int c[] __attribute__ ((counted_by (b)));
+} *array_nested_annotated;
+
+void __attribute__((__noinline__)) setup (int normal_count, int attr_count)
+{
+ array_flex
+ = (struct flex *)malloc (sizeof (struct flex)
+ + normal_count * sizeof (int));
+ array_flex->b = normal_count;
+
+ array_annotated
+ = (struct annotated *)malloc (sizeof (struct annotated)
+ + attr_count * sizeof (int));
+ array_annotated->b = attr_count;
+
+ array_nested_annotated
+ = (struct nested_annotated *)malloc (sizeof (struct nested_annotated)
+ + attr_count * sizeof (int));
+ array_nested_annotated->b = attr_count;
+
+ return;
+}
+
+void __attribute__((__noinline__)) test ()
+{
+ expect(__builtin_dynamic_object_size(array_flex->c, 1), -1);
+ expect(__builtin_dynamic_object_size(array_annotated->c, 1),
+ array_annotated->b * sizeof (int));
+ expect(__builtin_dynamic_object_size(array_nested_annotated->c, 1),
+ array_nested_annotated->b * sizeof (int));
+}
+
+int main(int argc, char *argv[])
+{
+ setup (10,10);
+ test ();
+ DONE ();
+}
diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
new file mode 100644
index 000000000000..a0c3cb88ec71
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
@@ -0,0 +1,210 @@
+/* test the attribute counted_by and its usage in
+__builtin_dynamic_object_size: what's the correct behavior when the
+allocation size mismatched with the value of counted_by attribute? */
+/* { dg-do run } */
+/* { dg-options "-O -fstrict-flex-arrays=3" } */
+
+#include "builtin-object-size-common.h"
+
+struct annotated {
+ size_t foo;
+ char others;
+ char array[] __attribute__((counted_by (foo)));
+};
+
+#define expect(p, _v) do { \
+ size_t v = _v; \
+ if (p == v) \
+ __builtin_printf ("ok: %s == %zd\n", #p, p); \
+ else \
+ { \
+ __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
+ FAIL (); \
+ } \
+} while (0);
+
+#define noinline __attribute__((__noinline__))
+#define SIZE_BUMP 10
+#define MAX(a, b) ((a) > (b) ? (a) : (b))
+#define MIN(a, b) ((a) < (b) ? (a) : (b))
+
+/* In general, Due to type casting, the type for the pointee of a pointer
+ does not say anything about the object it points to,
+ So, __builtin_object_size can not directly use the type of the pointee
+ to decide the size of the object the pointer points to.
+
+ there are only two reliable ways:
+ A. observed allocations (call to the allocation functions in the routine)
+ B. observed accesses (read or write access to the location of the
+ pointer points to)
+
+ that provide information about the type/existence of an object at
+ the corresponding address.
+
+ for A, we use the "alloc_size" attribute for the corresponding allocation
+ functions to determine the object size;
+
+ For B, we use the SIZE info of the TYPE attached to the corresponding access.
+ (We treat counted_by attribute as a complement to the SIZE info of the TYPE
+ for FMA)
+
+ The only other way in C which ensures that a pointer actually points
+ to an object of the correct type is 'static':
+
+ void foo(struct P *p[static 1]);
+
+ See https://gcc.gnu.org/pipermail/gcc-patches/2023-July/624814.html
+ for more details. */
+
+/* in the following function, malloc allocated more space than the value
+ of counted_by attribute. Then what's the correct behavior we expect
+ the __builtin_dynamic_object_size should have for each of the cases? */
+
+static struct annotated * noinline alloc_buf_more (size_t index)
+{
+ struct annotated *p;
+ size_t allocated_size
+ = MAX (sizeof (struct annotated),
+ (__builtin_offsetof (struct annotated, array[0])
+ + (index + SIZE_BUMP) * sizeof (char)));
+ p = (struct annotated *) malloc (allocated_size);
+
+ p->foo = index;
+
+ /*when checking the observed access p->array, we have info on both
+ observered allocation and observed access,
+ A. from observed allocation:
+ allocated_size - offsetof (struct annotated, array[0])
+ B. from observed access: p->foo * sizeof (char)
+ */
+
+ /* for size in the whole object: always uses A. */
+ /* for size in the sub-object: chose the smaller of A and B.
+ * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
+ * for details on why. */
+
+ /* for MAXIMUM size in the whole object: use the allocation size
+ for the whole object. */
+ expect(__builtin_dynamic_object_size(p->array, 0),
+ allocated_size - __builtin_offsetof (struct annotated, array[0]));
+
+ /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
+ expect(__builtin_dynamic_object_size(p->array, 1),
+ MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
+ (p->foo) * sizeof(char)));
+
+ /* for MINIMUM size in the whole object: use the allocation size
+ for the whole object. */
+ expect(__builtin_dynamic_object_size(p->array, 2),
+ allocated_size - __builtin_offsetof (struct annotated, array[0]));
+
+ /* for MINIMUM size in the sub-object: use the smaller of A and B. */
+ expect(__builtin_dynamic_object_size(p->array, 3),
+ MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
+ (p->foo) * sizeof(char)));
+
+ /*when checking the pointer p, we only have info on the observed allocation.
+ So, the object size info can only been obtained from the call to malloc.
+ for both MAXIMUM and MINIMUM: A = (index + SIZE_BUMP) * sizeof (char) */
+ expect(__builtin_dynamic_object_size(p, 0), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 1), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 2), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 3), allocated_size);
+ return p;
+}
+
+/* in the following function, malloc allocated less space than the value
+ of counted_by attribute. Then what's the correct behavior we expect
+ the __builtin_dynamic_object_size should have for each of the cases?
+ NOTE: this is an user error, GCC should issue warnings for such case.
+ this is a seperate issue we should address later. */
+
+static struct annotated * noinline alloc_buf_less (size_t index)
+{
+ struct annotated *p;
+ size_t allocated_size
+ = MAX (sizeof (struct annotated),
+ (__builtin_offsetof (struct annotated, array[0])
+ + (index) * sizeof (char)));
+ p = (struct annotated *) malloc (allocated_size);
+
+ p->foo = index + SIZE_BUMP;
+
+ /*when checking the observed access p->array, we have info on both
+ observered allocation and observed access,
+ A. from observed allocation:
+ allocated_size - offsetof (struct annotated, array[0])
+ B. from observed access: p->foo * sizeof (char)
+ */
+
+ /* for size in the whole object: always uses A. */
+ /* for size in the sub-object: chose the smaller of A and B.
+ * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
+ * for details on why. */
+
+ /* for MAXIMUM size in the whole object: use the allocation size
+ for the whole object. */
+ expect(__builtin_dynamic_object_size(p->array, 0),
+ allocated_size - __builtin_offsetof (struct annotated, array[0]));
+
+ /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
+ expect(__builtin_dynamic_object_size(p->array, 1),
+ MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
+ (p->foo) * sizeof(char)));
+
+ /* for MINIMUM size in the whole object: use the allocation size
+ for the whole object. */
+ expect(__builtin_dynamic_object_size(p->array, 2),
+ allocated_size - __builtin_offsetof (struct annotated, array[0]));
+
+ /* for MINIMUM size in the sub-object: use the smaller of A and B. */
+ expect(__builtin_dynamic_object_size(p->array, 3),
+ MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
+ (p->foo) * sizeof(char)));
+
+ /*when checking the pointer p, we only have info on the observed
+ allocation. So, the object size info can only been obtained from
+ the call to malloc. */
+ expect(__builtin_dynamic_object_size(p, 0), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 1), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 2), allocated_size);
+ expect(__builtin_dynamic_object_size(p, 3), allocated_size);
+ return p;
+}
+
+int main ()
+{
+ struct annotated *p, *q;
+ p = alloc_buf_more (10);
+ q = alloc_buf_less (10);
+
+ /*when checking the observed access p->array, we only have info on the
+ observed access, i.e, the TYPE_SIZE info from the access. We don't have
+ info on the whole object. */
+ expect(__builtin_dynamic_object_size(p->array, 0), -1);
+ expect(__builtin_dynamic_object_size(p->array, 1), p->foo * sizeof(char));
+ expect(__builtin_dynamic_object_size(p->array, 2), 0);
+ expect(__builtin_dynamic_object_size(p->array, 3), p->foo * sizeof(char));
+ /*when checking the pointer p, we have no observed allocation nor observed
+ access, therefore, we cannot determine the size info here. */
+ expect(__builtin_dynamic_object_size(p, 0), -1);
+ expect(__builtin_dynamic_object_size(p, 1), -1);
+ expect(__builtin_dynamic_object_size(p, 2), 0);
+ expect(__builtin_dynamic_object_size(p, 3), 0);
+
+ /*when checking the observed access p->array, we only have info on the
+ observed access, i.e, the TYPE_SIZE info from the access. We don't have
+ info on the whole object. */
+ expect(__builtin_dynamic_object_size(q->array, 0), -1);
+ expect(__builtin_dynamic_object_size(q->array, 1), q->foo * sizeof(char));
+ expect(__builtin_dynamic_object_size(q->array, 2), 0);
+ expect(__builtin_dynamic_object_size(q->array, 3), q->foo * sizeof(char));
+ /*when checking the pointer p, we have no observed allocation nor observed
+ access, therefore, we cannot determine the size info here. */
+ expect(__builtin_dynamic_object_size(q, 0), -1);
+ expect(__builtin_dynamic_object_size(q, 1), -1);
+ expect(__builtin_dynamic_object_size(q, 2), 0);
+ expect(__builtin_dynamic_object_size(q, 3), 0);
+
+ DONE ();
+}
diff --git a/gcc/tree-object-size.cc b/gcc/tree-object-size.cc
index a62af0500563..cf7843c5684b 100644
--- a/gcc/tree-object-size.cc
+++ b/gcc/tree-object-size.cc
@@ -585,6 +585,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
if (pt_var != TREE_OPERAND (ptr, 0))
{
tree var;
+ tree counted_by_ref = NULL_TREE;
if (object_size_type & OST_SUBOBJECT)
{
@@ -600,11 +601,12 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
var = TREE_OPERAND (var, 0);
if (var != pt_var && TREE_CODE (var) == ARRAY_REF)
var = TREE_OPERAND (var, 0);
- if (! TYPE_SIZE_UNIT (TREE_TYPE (var))
+ if (! component_ref_has_counted_by_p (var)
+ && ((! TYPE_SIZE_UNIT (TREE_TYPE (var))
|| ! tree_fits_uhwi_p (TYPE_SIZE_UNIT (TREE_TYPE (var)))
|| (pt_var_size && TREE_CODE (pt_var_size) == INTEGER_CST
&& tree_int_cst_lt (pt_var_size,
- TYPE_SIZE_UNIT (TREE_TYPE (var)))))
+ TYPE_SIZE_UNIT (TREE_TYPE (var)))))))
var = pt_var;
else if (var != pt_var && TREE_CODE (pt_var) == MEM_REF)
{
@@ -612,6 +614,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
/* For &X->fld, compute object size if fld isn't a flexible array
member. */
bool is_flexible_array_mem_ref = false;
+
while (v && v != pt_var)
switch (TREE_CODE (v))
{
@@ -660,6 +663,8 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
/* Now the ref is to an array type. */
gcc_assert (TREE_CODE (TREE_TYPE (v)) == ARRAY_TYPE);
is_flexible_array_mem_ref = array_ref_flexible_size_p (v);
+ counted_by_ref = component_ref_get_counted_by (v);
+
while (v != pt_var && TREE_CODE (v) == COMPONENT_REF)
if (TREE_CODE (TREE_TYPE (TREE_OPERAND (v, 0)))
!= UNION_TYPE
@@ -673,8 +678,11 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
== RECORD_TYPE)
{
/* compute object size only if v is not a
- flexible array member. */
- if (!is_flexible_array_mem_ref)
+ flexible array member or the flexible array member
+ has a known element count indicated by the user
+ through attribute counted_by. */
+ if (!is_flexible_array_mem_ref
+ || counted_by_ref)
{
v = NULL_TREE;
break;
@@ -707,9 +715,24 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
if (var != pt_var)
{
- var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
- if (!TREE_CONSTANT (var_size))
- var_size = get_or_create_ssa_default_def (cfun, var_size);
+ if (!counted_by_ref)
+ {
+ var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
+ if (!TREE_CONSTANT (var_size))
+ var_size = get_or_create_ssa_default_def (cfun, var_size);
+ }
+ else
+ {
+ gcc_assert (TREE_CODE (var) == COMPONENT_REF
+ && TREE_CODE (TREE_TYPE (var)) == ARRAY_TYPE);
+ tree element_size = TYPE_SIZE_UNIT (TREE_TYPE (TREE_TYPE (var)));
+ var_size
+ = size_binop (MULT_EXPR,
+ fold_convert (sizetype, counted_by_ref),
+ fold_convert (sizetype, element_size));
+ if (!todo)
+ todo = TODO_update_ssa_only_virtuals;
+ }
if (!var_size)
return false;
}
diff --git a/gcc/tree.cc b/gcc/tree.cc
index fcd36ae0cd74..3b6ddcbdcbf8 100644
--- a/gcc/tree.cc
+++ b/gcc/tree.cc
@@ -12745,6 +12745,32 @@ array_ref_element_size (tree exp)
return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
}
+/* For a component_ref that has an array type ARRAY_REF, return TRUE when
+ an counted_by attribute attached to the corresponding FIELD_DECL.
+ return FALSE otherwise. */
+bool
+component_ref_has_counted_by_p (tree array_ref)
+{
+ if (TREE_CODE (array_ref) != COMPONENT_REF)
+ return false;
+
+ if (TREE_CODE (TREE_TYPE (array_ref)) != ARRAY_TYPE)
+ return false;
+
+ tree struct_object = TREE_OPERAND (array_ref, 0);
+ tree struct_type = TREE_TYPE (struct_object);
+
+ if (!RECORD_OR_UNION_TYPE_P (struct_type))
+ return false;
+ tree field_decl = TREE_OPERAND (array_ref, 1);
+ tree attr_counted_by = lookup_attribute ("counted_by",
+ DECL_ATTRIBUTES (field_decl));
+
+ if (!attr_counted_by)
+ return false;
+ return true;
+}
+
/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
whose name is FIELDNAME, which is the last TREE_VALUE of the list.
@@ -12771,7 +12797,7 @@ get_named_field (tree fieldlist, const char *fieldname)
fields inside it recursively. */
else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
- fieldname)) != NULL_TREE)
+ fieldname)) != NULL_TREE)
{
named_field = tree_cons (NULL_TREE, field, named_field);
break;
@@ -12784,6 +12810,73 @@ get_named_field (tree fieldlist, const char *fieldname)
return named_field;
}
+/* For a component_ref that has an array type ARRAY_REF, get the object that
+ represents its counted_by per the attribute counted_by attached to
+ the corresponding FIELD_DECL. return NULL_TREE when cannot find such
+ object.
+ For example, if:
+
+ struct P {
+ int k;
+ int x[] __attribute__ ((counted_by (k)));
+ } *p;
+
+ for the following reference:
+
+ p->x[b]
+
+ the object that represents its element count will be:
+
+ p->k
+
+ So, when component_ref_get_counted_by (p->x[b]) is called, p->k should be
+ returned.
+*/
+
+tree
+component_ref_get_counted_by (tree array_ref)
+{
+ if (! component_ref_has_counted_by_p (array_ref))
+ return NULL_TREE;
+
+ tree struct_object = TREE_OPERAND (array_ref, 0);
+ tree struct_type = TREE_TYPE (struct_object);
+ tree field_decl = TREE_OPERAND (array_ref, 1);
+ tree attr_counted_by = lookup_attribute ("counted_by",
+ DECL_ATTRIBUTES (field_decl));
+ gcc_assert (attr_counted_by);
+
+ /* If there is an counted_by attribute attached to the field,
+ get the field that maps to the counted_by. */
+
+ const char *fieldname
+ = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
+
+ tree counted_by_field = get_named_field (TYPE_FIELDS (struct_type),
+ fieldname);
+
+ gcc_assert (counted_by_field);
+
+ /* generate the tree node that represent the counted_by of this array
+ ref. This is a (possible nested) COMPONENT_REF to the counted_by_field
+ of the containing structure. */
+
+ tree counted_by_ref = NULL_TREE;
+ tree object = struct_object;
+ do
+ {
+ tree field = TREE_VALUE (counted_by_field);
+
+ counted_by_ref = build3 (COMPONENT_REF,
+ TREE_TYPE (field),
+ unshare_expr (object), field,
+ NULL_TREE);
+ object = counted_by_ref;
+ counted_by_field = TREE_CHAIN (counted_by_field);
+ }
+ while (counted_by_field);
+ return counted_by_ref;
+}
/* Return a tree representing the lower bound of the array mentioned in
EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
diff --git a/gcc/tree.h b/gcc/tree.h
index 4859becaa1e7..07eed7219835 100644
--- a/gcc/tree.h
+++ b/gcc/tree.h
@@ -5619,11 +5619,21 @@ extern tree get_base_address (tree t);
of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
extern tree array_ref_element_size (tree);
+/* Give a component_ref that has an array type, return true when an
+ attribute counted_by attached to the corresponding FIELD_DECL. */
+extern bool component_ref_has_counted_by_p (tree);
+
/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
name is FIELDNAME, return NULL_TREE if such field is not found.
searching nested anonymous structure/union recursively. */
extern tree get_named_field (tree, const char *);
+/* Give a component_ref that has an array type, return the object that
+ represents its counted_by per the attribute counted_by attached to
+ the corresponding FIELD_DECL. return NULL_TREE when cannot find such
+ object. */
+extern tree component_ref_get_counted_by (tree);
+
/* Return a typenode for the "standard" C type with a given name. */
extern tree get_typenode_from_name (const char *);
--
2.31.1
^ permalink raw reply [flat|nested] 116+ messages in thread
* [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896]
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
2023-08-25 15:24 ` [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896] Qing Zhao
@ 2023-08-25 15:24 ` Qing Zhao
2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:45 ` PING * 2: " Qing Zhao
2023-08-25 19:51 ` [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Kees Cook
` (3 subsequent siblings)
6 siblings, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-08-25 15:24 UTC (permalink / raw)
To: joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard, Qing Zhao
Use the counted_by attribute information in bound sanitizer.
gcc/c-family/ChangeLog:
PR C/108896
* c-ubsan.cc (ubsan_instrument_bounds): Use counted_by attribute
information.
gcc/testsuite/ChangeLog:
PR C/108896
* gcc.dg/ubsan/flex-array-counted-by-bounds.c: New test.
* gcc.dg/ubsan/flex-array-counted-by-bounds-2.c: New test.
---
gcc/c-family/c-ubsan.cc | 16 +++++++
.../ubsan/flex-array-counted-by-bounds-2.c | 27 +++++++++++
.../ubsan/flex-array-counted-by-bounds.c | 46 +++++++++++++++++++
3 files changed, 89 insertions(+)
create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
diff --git a/gcc/c-family/c-ubsan.cc b/gcc/c-family/c-ubsan.cc
index 51aa83a378d2..a99e8433069f 100644
--- a/gcc/c-family/c-ubsan.cc
+++ b/gcc/c-family/c-ubsan.cc
@@ -362,6 +362,10 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
{
tree type = TREE_TYPE (array);
tree domain = TYPE_DOMAIN (type);
+ /* whether the array ref is a flexible array member with valid counted_by
+ attribute. */
+ bool fam_has_count_attr = false;
+ tree counted_by = NULL_TREE;
if (domain == NULL_TREE)
return NULL_TREE;
@@ -375,6 +379,17 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
&& COMPLETE_TYPE_P (type)
&& integer_zerop (TYPE_SIZE (type)))
bound = build_int_cst (TREE_TYPE (TYPE_MIN_VALUE (domain)), -1);
+ /* If the array ref is to flexible array member field which has
+ counted_by attribute. We can use the information from the
+ attribute as the bound to instrument the reference. */
+ else if ((counted_by = component_ref_get_counted_by (array))
+ != NULL_TREE)
+ {
+ fam_has_count_attr = true;
+ bound = fold_build2 (MINUS_EXPR, TREE_TYPE (counted_by),
+ counted_by,
+ build_int_cst (TREE_TYPE (counted_by), 1));
+ }
else
return NULL_TREE;
}
@@ -387,6 +402,7 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
-fsanitize=bounds-strict. */
tree base = get_base_address (array);
if (!sanitize_flags_p (SANITIZE_BOUNDS_STRICT)
+ && !fam_has_count_attr
&& TREE_CODE (array) == COMPONENT_REF
&& base && (INDIRECT_REF_P (base) || TREE_CODE (base) == MEM_REF))
{
diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
new file mode 100644
index 000000000000..77ec333509d0
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
@@ -0,0 +1,27 @@
+/* test the attribute counted_by and its usage in
+ bounds sanitizer combined with VLA. */
+/* { dg-do run } */
+/* { dg-options "-fsanitize=bounds" } */
+
+#include <stdlib.h>
+
+void __attribute__((__noinline__)) setup_and_test_vla (int n, int m)
+{
+ struct foo {
+ int n;
+ int p[][n] __attribute__((counted_by(n)));
+ } *f;
+
+ f = (struct foo *) malloc (sizeof(struct foo) + m*sizeof(int[n]));
+ f->n = m;
+ f->p[m][n-1]=1;
+ return;
+}
+
+int main(int argc, char *argv[])
+{
+ setup_and_test_vla (10, 11);
+ return 0;
+}
+
+/* { dg-output "17:8: runtime error: index 11 out of bounds for type" } */
diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
new file mode 100644
index 000000000000..81eaeb3f2681
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
@@ -0,0 +1,46 @@
+/* test the attribute counted_by and its usage in
+ bounds sanitizer. */
+/* { dg-do run } */
+/* { dg-options "-fsanitize=bounds" } */
+
+#include <stdlib.h>
+
+struct flex {
+ int b;
+ int c[];
+} *array_flex;
+
+struct annotated {
+ int b;
+ int c[] __attribute__ ((counted_by (b)));
+} *array_annotated;
+
+void __attribute__((__noinline__)) setup (int normal_count, int annotated_count)
+{
+ array_flex
+ = (struct flex *)malloc (sizeof (struct flex)
+ + normal_count * sizeof (int));
+ array_flex->b = normal_count;
+
+ array_annotated
+ = (struct annotated *)malloc (sizeof (struct annotated)
+ + annotated_count * sizeof (int));
+ array_annotated->b = annotated_count;
+
+ return;
+}
+
+void __attribute__((__noinline__)) test (int normal_index, int annotated_index)
+{
+ array_flex->c[normal_index] = 1;
+ array_annotated->c[annotated_index] = 2;
+}
+
+int main(int argc, char *argv[])
+{
+ setup (10, 10);
+ test (10, 10);
+ return 0;
+}
+
+/* { dg-output "36:21: runtime error: index 10 out of bounds for type" } */
--
2.31.1
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
` (2 preceding siblings ...)
2023-08-25 15:24 ` [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896] Qing Zhao
@ 2023-08-25 19:51 ` Kees Cook
2023-09-08 14:11 ` Qing Zhao
` (2 subsequent siblings)
6 siblings, 0 replies; 116+ messages in thread
From: Kees Cook @ 2023-08-25 19:51 UTC (permalink / raw)
To: Qing Zhao
Cc: joseph, richard.guenther, jakub, gcc-patches, siddhesh, uecker, isanbard
On Fri, Aug 25, 2023 at 03:24:22PM +0000, Qing Zhao wrote:
> This is the 3rd version of the patch, per our discussion based on the
> review comments for the 1st and 2nd version, the major changes in this
This tests out great for me; thanks you! I'm able to build the entire
kernel tree with 201 annotations[1] added. Things work as expected. :)
-Kees
[1] https://git.kernel.org/pub/scm/linux/kernel/git/kees/linux.git/log/?h=devel/next-20230825/counted_by
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
` (3 preceding siblings ...)
2023-08-25 19:51 ` [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Kees Cook
@ 2023-09-08 14:11 ` Qing Zhao
2023-09-20 13:43 ` PING * 2: " Qing Zhao
2023-10-05 20:08 ` Siddhesh Poyarekar
6 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-08 14:11 UTC (permalink / raw)
To: Joseph Myers, Richard Biener, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Ping.
Thanks.
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> This is the 3rd version of the patch, per our discussion based on the
> review comments for the 1st and 2nd version, the major changes in this
> version are:
>
> ***Against 1st version:
> 1. change the name "element_count" to "counted_by";
> 2. change the parameter for the attribute from a STRING to an
> Identifier;
> 3. Add logic and testing cases to handle anonymous structure/unions;
> 4. Clarify documentation to permit the situation when the allocation
> size is larger than what's specified by "counted_by", at the same time,
> it's user's error if allocation size is smaller than what's specified by
> "counted_by";
> 5. Add a complete testing case for using counted_by attribute in
> __builtin_dynamic_object_size when there is mismatch between the
> allocation size and the value of "counted_by", the expecting behavior
> for each case and the explanation on why in the comments.
>
> ***Against 2rd version:
> 1. Identify a tree node sharing issue and fixed it in the routine
> "component_ref_get_counted_ty" of tree.cc;
> 2. Update the documentation and testing cases with the clear usage
> of the fomula to compute the allocation size:
> MAX (sizeof (struct A), offsetof (struct A, array[0]) + counted_by * sizeof(element))
> (the algorithm used in tree-object-size.cc is correct).
>
> In this set of patches, the major functionality provided is:
>
> 1. a new attribute "counted_by";
> 2. use this new attribute in bound sanitizer;
> 3. use this new attribute in dynamic object size for subobject size;
>
> As discussed, I plan to add two more separate patches sets after this initial
> patch set is approved and committed.
>
> set 1. A new warning option and a new sanitizer option for the user error
> when the allocation size is smaller than the value of "counted_by".
> set 2. An improvement to __builtin_dynamic_object_size for whole-object
> size of the structure with FAM annaoted with counted_by.
>
> there are also some existing bugs in tree-object-size.cc identified
> during the study, and PRs were filed to record them. these bugs will
> be fixed seperately with individual patches:
>
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111030
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111040
>
> Bootstrapped and regression tested on both aarch64 and X86, no issue.
>
> Please see more details on the description of this work on:
>
> https://gcc.gnu.org/pipermail/gcc-patches/2023-May/619708.html
>
> and more discussions on
> https://gcc.gnu.org/pipermail/gcc-patches/2023-August/626376.html
>
> Okay for committing?
>
> thanks.
>
> Qing
>
> Qing Zhao (3):
> Provide counted_by attribute to flexible array member field (PR108896)
> Use the counted_by atribute info in builtin object size [PR108896]
> Use the counted_by attribute information in bound sanitizer[PR108896]
>
> gcc/c-family/c-attribs.cc | 54 ++++-
> gcc/c-family/c-common.cc | 13 ++
> gcc/c-family/c-common.h | 1 +
> gcc/c-family/c-ubsan.cc | 16 ++
> gcc/c/c-decl.cc | 79 +++++--
> gcc/doc/extend.texi | 77 +++++++
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++
> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++
> .../ubsan/flex-array-counted-by-bounds.c | 46 ++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 133 +++++++++++
> gcc/tree.h | 15 ++
> 14 files changed, 797 insertions(+), 25 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
@ 2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:44 ` Ping * 2: " Qing Zhao
2023-10-05 18:51 ` Siddhesh Poyarekar
2 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-08 14:12 UTC (permalink / raw)
To: Joseph Myers, richard.guenther, jakub Jelinek, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
PIng.
thanks.
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Provide a new counted_by attribute to flexible array member field.
>
> 'counted_by (COUNT)'
> The 'counted_by' attribute may be attached to the flexible array
> member of a structure. It indicates that the number of the
> elements of the array is given by the field named "COUNT" in the
> same structure as the flexible array member. GCC uses this
> information to improve the results of the array bound sanitizer and
> the '__builtin_dynamic_object_size'.
>
> For instance, the following code:
>
> struct P {
> size_t count;
> char other;
> char array[] __attribute__ ((counted_by (count)));
> } *p;
>
> specifies that the 'array' is a flexible array member whose number
> of elements is given by the field 'count' in the same structure.
>
> The field that represents the number of the elements should have an
> integer type. An explicit 'counted_by' annotation defines a
> relationship between two objects, 'p->array' and 'p->count', that
> 'p->array' has _at least_ 'p->count' number of elements available.
> This relationship must hold even after any of these related objects
> are updated. It's the user's responsibility to make sure this
> relationship to be kept all the time. Otherwise the results of the
> array bound sanitizer and the '__builtin_dynamic_object_size' might
> be incorrect.
>
> For instance, in the following example, the allocated array has
> less elements than what's specified by the 'sbuf->count', this is
> an user error. As a result, out-of-bounds access to the array
> might not be detected.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + nelems * sizeof (char))));
> sbuf->count = nelems + SIZE_BUMP;
> /* This is invalid when the sbuf->array has less than sbuf->count
> elements. */
> }
>
> In the following example, the 2nd update to the field 'sbuf->count'
> of the above structure will permit out-of-bounds access to the
> array 'sbuf>array' as well.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + (nelems + SIZE_BUMP) * sizeof (char))));
> sbuf->count = nelems;
> /* This is valid when the sbuf->array has at least sbuf->count
> elements. */
> }
> void use_buf (int index)
> {
> sbuf->count = sbuf->count + SIZE_BUMP + 1;
> /* Now the value of sbuf->count is larger than the number
> of elements of sbuf->array. */
> sbuf->array[index] = 0;
> /* then the out-of-bound access to this array
> might not be detected. */
> }
>
> gcc/c-family/ChangeLog:
>
> PR C/108896
> * c-attribs.cc (handle_counted_by_attribute): New function.
> (attribute_takes_identifier_p): Add counted_by attribute to the list.
> * c-common.cc (c_flexible_array_member_type_p): ...To this.
> * c-common.h (c_flexible_array_member_type_p): New prototype.
>
> gcc/c/ChangeLog:
>
> PR C/108896
> * c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
> (add_flexible_array_elts_to_size): Use renamed function.
> (is_flexible_array_member_p): Use renamed function.
> (verify_counted_by_attribute): New function.
> (finish_struct): Use renamed function and verify counted_by
> attribute.
>
> gcc/ChangeLog:
>
> PR C/108896
> * doc/extend.texi: Document attribute counted_by.
> * tree.cc (get_named_field): New function.
> * tree.h (get_named_field): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by.c: New test.
> ---
> gcc/c-family/c-attribs.cc | 54 ++++++++++++-
> gcc/c-family/c-common.cc | 13 ++++
> gcc/c-family/c-common.h | 1 +
> gcc/c/c-decl.cc | 79 +++++++++++++++-----
> gcc/doc/extend.texi | 77 +++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
> gcc/tree.cc | 40 ++++++++++
> gcc/tree.h | 5 ++
> 8 files changed, 291 insertions(+), 18 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>
> diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
> index e2792ca6898b..65e4f6639109 100644
> --- a/gcc/c-family/c-attribs.cc
> +++ b/gcc/c-family/c-attribs.cc
> @@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute (tree *, tree, tree,
> int, bool *);
> static tree handle_strict_flex_array_attribute (tree *, tree, tree,
> int, bool *);
> +static tree handle_counted_by_attribute (tree *, tree, tree,
> + int, bool *);
> static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree, bool *);
> @@ -373,6 +375,8 @@ const struct attribute_spec c_common_attribute_table[] =
> handle_warn_if_not_aligned_attribute, NULL },
> { "strict_flex_array", 1, 1, true, false, false, false,
> handle_strict_flex_array_attribute, NULL },
> + { "counted_by", 1, 1, true, false, false, false,
> + handle_counted_by_attribute, NULL },
> { "weak", 0, 0, true, false, false, false,
> handle_weak_attribute, NULL },
> { "noplt", 0, 0, true, false, false, false,
> @@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
> else if (!strcmp ("mode", spec->name)
> || !strcmp ("format", spec->name)
> || !strcmp ("cleanup", spec->name)
> - || !strcmp ("access", spec->name))
> + || !strcmp ("access", spec->name)
> + || !strcmp ("counted_by", spec->name))
> return true;
> else
> return targetm.attribute_takes_identifier_p (attr_id);
> @@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node, tree name,
> return NULL_TREE;
> }
>
> +/* Handle a "counted_by" attribute; arguments as in
> + struct attribute_spec.handler. */
> +
> +static tree
> +handle_counted_by_attribute (tree *node, tree name,
> + tree args, int ARG_UNUSED (flags),
> + bool *no_add_attrs)
> +{
> + tree decl = *node;
> + tree argval = TREE_VALUE (args);
> +
> + /* This attribute only applies to field decls of a structure. */
> + if (TREE_CODE (decl) != FIELD_DECL)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for non-field"
> + " declaration %q+D", name, decl);
> + *no_add_attrs = true;
> + }
> + /* This attribute only applies to field with array type. */
> + else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non-array field",
> + name);
> + *no_add_attrs = true;
> + }
> + /* This attribute only applies to a C99 flexible array member type. */
> + else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non"
> + " flexible array member field",
> + name);
> + *no_add_attrs = true;
> + }
> + /* The argument should be an identifier. */
> + else if (TREE_CODE (argval) != IDENTIFIER_NODE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%<counted_by%> argument not an identifier");
> + *no_add_attrs = true;
> + }
> +
> + return NULL_TREE;
> +}
> +
> /* Handle a "weak" attribute; arguments as in
> struct attribute_spec.handler. */
>
> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
> index 9fbaeb437a12..a18937245c2a 100644
> --- a/gcc/c-family/c-common.cc
> +++ b/gcc/c-family/c-common.cc
> @@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
> (*debug_hooks->early_global_decl) (cnode->decl);
> }
>
> +/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> +bool
> +c_flexible_array_member_type_p (const_tree type)
> +{
> + if (TREE_CODE (type) == ARRAY_TYPE
> + && TYPE_SIZE (type) == NULL_TREE
> + && TYPE_DOMAIN (type) != NULL_TREE
> + && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> + return true;
> +
> + return false;
> +}
> +
> /* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based on the
> values of attribute strict_flex_array and the flag_strict_flex_arrays. */
> unsigned int
> diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
> index 78fc5248ba68..c29bb429062b 100644
> --- a/gcc/c-family/c-common.h
> +++ b/gcc/c-family/c-common.h
> @@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
> extern tree c_common_get_narrower (tree, int *);
> extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
> extern void c_common_finalize_early_debug (void);
> +extern bool c_flexible_array_member_type_p (const_tree);
> extern unsigned int c_strict_flex_array_level_of (tree);
> extern bool c_option_is_from_cpp_diagnostics (int);
>
> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
> index 1f9eb44dbaa2..e943b49b5230 100644
> --- a/gcc/c/c-decl.cc
> +++ b/gcc/c/c-decl.cc
> @@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator *decl,
> return decl;
> }
>
> -/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> -static bool
> -flexible_array_member_type_p (const_tree type)
> -{
> - if (TREE_CODE (type) == ARRAY_TYPE
> - && TYPE_SIZE (type) == NULL_TREE
> - && TYPE_DOMAIN (type) != NULL_TREE
> - && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> - return true;
> -
> - return false;
> -}
> -
> /* Determine whether TYPE is a one-element array type "[1]". */
> static bool
> one_element_array_type_p (const_tree type)
> @@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree init)
>
> elt = CONSTRUCTOR_ELTS (init)->last ().value;
> type = TREE_TYPE (elt);
> - if (flexible_array_member_type_p (type))
> + if (c_flexible_array_member_type_p (type))
> {
> complete_array_type (&type, elt, false);
> DECL_SIZE (decl)
> @@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
>
> bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
> bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
> - bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
> + bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE (x));
>
> unsigned int strict_flex_array_level = c_strict_flex_array_level_of (x);
>
> @@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
> return false;
> }
>
> +/* Verify the argument of the counted_by attribute of the flexible array
> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
> + FIELDLIST, Report error and remove this attribute when it's not. */
> +static void
> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
> +{
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return;
> +
> + /* If there is an counted_by attribute attached to the field,
> + verify it. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + /* Verify the argument of the attrbute is a valid field of the
> + containing structure. */
> +
> + tree counted_by_field = get_named_field (fieldlist, fieldname);
> +
> + /* Error when the field is not found in the containing structure. */
> + if (!counted_by_field)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " in the same structure, ignore it",
> + (get_attribute_name (attr_counted_by)));
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + else
> + /* Error when the field is not with an integer type. */
> + {
> + while (TREE_CHAIN (counted_by_field))
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + tree real_field = TREE_VALUE (counted_by_field);
> +
> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " with integer type, ignore it",
> + (get_attribute_name (attr_counted_by)));
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + }
> +
> + return;
> +}
>
> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
> LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
> @@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> DECL_PACKED (x) = 1;
>
> /* Detect flexible array member in an invalid context. */
> - if (flexible_array_member_type_p (TREE_TYPE (x)))
> + if (c_flexible_array_member_type_p (TREE_TYPE (x)))
> {
> if (TREE_CODE (t) == UNION_TYPE)
> {
> @@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> "members");
> TREE_TYPE (x) = error_mark_node;
> }
> + /* if there is a counted_by attribute attached to this field,
> + verify it. */
> + verify_counted_by_attribute (fieldlist, x);
> }
>
> if (pedantic && TREE_CODE (t) == RECORD_TYPE
> @@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> when x is an array and is the last field. */
> if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
> TYPE_INCLUDES_FLEXARRAY (t)
> - = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
> + = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
> /* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of x, t
> when x is an union or record and is the last field. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
> diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
> index 97eaacf8a7ec..ea6240646936 100644
> --- a/gcc/doc/extend.texi
> +++ b/gcc/doc/extend.texi
> @@ -7617,6 +7617,83 @@ When both the attribute and the option present at the same time, the level of
> the strictness for the specific trailing array field is determined by the
> attribute.
>
> +@cindex @code{counted_by} variable attribute
> +@item counted_by (@var{count})
> +The @code{counted_by} attribute may be attached to the flexible array
> +member of a structure. It indicates that the number of the elements of the
> +array is given by the field named "@var{count}" in the same structure as the
> +flexible array member. GCC uses this information to improve the results of
> +the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
> +
> +For instance, the following code:
> +
> +@smallexample
> +struct P @{
> + size_t count;
> + char other;
> + char array[] __attribute__ ((counted_by (count)));
> +@} *p;
> +@end smallexample
> +
> +@noindent
> +specifies that the @code{array} is a flexible array member whose number of
> +elements is given by the field @code{count} in the same structure.
> +
> +The field that represents the number of the elements should have an integer
> +type. An explicit @code{counted_by} annotation defines a relationship between
> +two objects, @code{p->array} and @code{p->count}, that @code{p->array} has
> +@emph{at least} @code{p->count} number of elements available. This relationship
> +must hold even after any of these related objects are updated. It's the user's
> +responsibility to make sure this relationship to be kept all the time.
> +Otherwise the results of the array bound sanitizer and the
> +@code{__builtin_dynamic_object_size} might be incorrect.
> +
> +For instance, in the following example, the allocated array has less elements
> +than what's specified by the @code{sbuf->count}, this is an user error. As a
> +result, out-of-bounds access to the array might not be detected.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + nelems * sizeof (char))));
> + sbuf->count = nelems + SIZE_BUMP;
> + /* This is invalid when the sbuf->array has less than sbuf->count
> + elements. */
> +@}
> +@end smallexample
> +
> +In the following example, the 2nd update to the field @code{sbuf->count} of
> +the above structure will permit out-of-bounds access to the array
> +@code{sbuf>array} as well.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + (nelems + SIZE_BUMP) * sizeof (char))));
> + sbuf->count = nelems;
> + /* This is valid when the sbuf->array has at least sbuf->count
> + elements. */
> +@}
> +void use_buf (int index)
> +@{
> + sbuf->count = sbuf->count + SIZE_BUMP + 1;
> + /* Now the value of sbuf->count is larger than the number
> + of elements of sbuf->array. */
> + sbuf->array[index] = 0;
> + /* then the out-of-bound access to this array
> + might not be detected. */
> +@}
> +@end smallexample
> +
> +
> @cindex @code{alloc_size} variable attribute
> @item alloc_size (@var{position})
> @itemx alloc_size (@var{position-1}, @var{position-2})
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> new file mode 100644
> index 000000000000..f8ce9776bf86
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> @@ -0,0 +1,40 @@
> +/* testing the correct usage of attribute counted_by. */
> +/* { dg-do compile } */
> +/* { dg-options "-O2" } */
> +
> +#include <wchar.h>
> +
> +int size;
> +int x __attribute ((counted_by (size))); /* { dg-error "attribute may not be specified for non-field declaration" } */
> +
> +struct trailing {
> + int count;
> + int field __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non-array field" } */
> +};
> +
> +struct trailing_1 {
> + int count;
> + int array_1[0] __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non flexible array member field" } */
> +};
> +
> +int count;
> +struct trailing_array_2 {
> + int count;
> + int array_2[] __attribute ((counted_by ("count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_3 {
> + int other;
> + int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_4 {
> + int other;
> + int array_4[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration in the same structure, ignore it" } */
> +};
> +
> +int count;
> +struct trailing_array_5 {
> + float count;
> + int array_5[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration with integer type, ignore it" } */
> +};
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index 420857b110c4..fcd36ae0cd74 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> + with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> + whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> + return NULL_TREE if such field is not found. Normally this list is of
> + length one, but if the field is embedded with (nested) anonymous structures
> + or unions, this list steps down the chain to the field. */
> +tree
> +get_named_field (tree fieldlist, const char *fieldname)
> +{
> + tree named_field = NULL_TREE;
> + for (tree field = fieldlist; field; field = DECL_CHAIN (field))
> + {
> + if (TREE_CODE (field) != FIELD_DECL)
> + continue;
> + if (DECL_NAME (field) != NULL)
> + if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + /* if the field is an anonymous struct/union, we will check the nested
> + fields inside it recursively. */
> + else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> + if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> + fieldname)) != NULL_TREE)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + else
> + continue;
> + }
> + return named_field;
> +}
> +
> +
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4c04245e2b1b..4859becaa1e7 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> + name is FIELDNAME, return NULL_TREE if such field is not found.
> + searching nested anonymous structure/union recursively. */
> +extern tree get_named_field (tree, const char *);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896]
2023-08-25 15:24 ` [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896] Qing Zhao
@ 2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:44 ` PING *2: " Qing Zhao
2023-10-05 20:01 ` Siddhesh Poyarekar
2 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-08 14:12 UTC (permalink / raw)
To: Joseph Myers, Richard Biener, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Ping.
thanks.
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Use the counted_by atribute info in builtin object size to compute the
> subobject size for flexible array members.
>
> gcc/ChangeLog:
>
> PR C/108896
> * tree-object-size.cc (addr_object_size): Use the counted_by
> attribute info.
> * tree.cc (component_ref_has_counted_by_p): New function.
> (component_ref_get_counted_by): New function.
> * tree.h (component_ref_has_counted_by_p): New prototype.
> (component_ref_get_counted_by): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by-2.c: New test.
> * gcc.dg/flex-array-counted-by-3.c: New test.
> ---
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 95 +++++++-
> gcc/tree.h | 10 +
> 5 files changed, 418 insertions(+), 8 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> new file mode 100644
> index 000000000000..ec580c1f1f01
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> @@ -0,0 +1,74 @@
> +/* test the attribute counted_by and its usage in
> + * __builtin_dynamic_object_size. */
> +/* { dg-do run } */
> +/* { dg-options "-O2" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
> +
> +struct flex {
> + int b;
> + int c[];
> +} *array_flex;
> +
> +struct annotated {
> + int b;
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_annotated;
> +
> +struct nested_annotated {
> + struct {
> + union {
> + int b;
> + float f;
> + };
> + int n;
> + };
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_nested_annotated;
> +
> +void __attribute__((__noinline__)) setup (int normal_count, int attr_count)
> +{
> + array_flex
> + = (struct flex *)malloc (sizeof (struct flex)
> + + normal_count * sizeof (int));
> + array_flex->b = normal_count;
> +
> + array_annotated
> + = (struct annotated *)malloc (sizeof (struct annotated)
> + + attr_count * sizeof (int));
> + array_annotated->b = attr_count;
> +
> + array_nested_annotated
> + = (struct nested_annotated *)malloc (sizeof (struct nested_annotated)
> + + attr_count * sizeof (int));
> + array_nested_annotated->b = attr_count;
> +
> + return;
> +}
> +
> +void __attribute__((__noinline__)) test ()
> +{
> + expect(__builtin_dynamic_object_size(array_flex->c, 1), -1);
> + expect(__builtin_dynamic_object_size(array_annotated->c, 1),
> + array_annotated->b * sizeof (int));
> + expect(__builtin_dynamic_object_size(array_nested_annotated->c, 1),
> + array_nested_annotated->b * sizeof (int));
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup (10,10);
> + test ();
> + DONE ();
> +}
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> new file mode 100644
> index 000000000000..a0c3cb88ec71
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> @@ -0,0 +1,210 @@
> +/* test the attribute counted_by and its usage in
> +__builtin_dynamic_object_size: what's the correct behavior when the
> +allocation size mismatched with the value of counted_by attribute? */
> +/* { dg-do run } */
> +/* { dg-options "-O -fstrict-flex-arrays=3" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +struct annotated {
> + size_t foo;
> + char others;
> + char array[] __attribute__((counted_by (foo)));
> +};
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
> +
> +#define noinline __attribute__((__noinline__))
> +#define SIZE_BUMP 10
> +#define MAX(a, b) ((a) > (b) ? (a) : (b))
> +#define MIN(a, b) ((a) < (b) ? (a) : (b))
> +
> +/* In general, Due to type casting, the type for the pointee of a pointer
> + does not say anything about the object it points to,
> + So, __builtin_object_size can not directly use the type of the pointee
> + to decide the size of the object the pointer points to.
> +
> + there are only two reliable ways:
> + A. observed allocations (call to the allocation functions in the routine)
> + B. observed accesses (read or write access to the location of the
> + pointer points to)
> +
> + that provide information about the type/existence of an object at
> + the corresponding address.
> +
> + for A, we use the "alloc_size" attribute for the corresponding allocation
> + functions to determine the object size;
> +
> + For B, we use the SIZE info of the TYPE attached to the corresponding access.
> + (We treat counted_by attribute as a complement to the SIZE info of the TYPE
> + for FMA)
> +
> + The only other way in C which ensures that a pointer actually points
> + to an object of the correct type is 'static':
> +
> + void foo(struct P *p[static 1]);
> +
> + See https://gcc.gnu.org/pipermail/gcc-patches/2023-July/624814.html
> + for more details. */
> +
> +/* in the following function, malloc allocated more space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases? */
> +
> +static struct annotated * noinline alloc_buf_more (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index + SIZE_BUMP) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed allocation.
> + So, the object size info can only been obtained from the call to malloc.
> + for both MAXIMUM and MINIMUM: A = (index + SIZE_BUMP) * sizeof (char) */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +/* in the following function, malloc allocated less space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases?
> + NOTE: this is an user error, GCC should issue warnings for such case.
> + this is a seperate issue we should address later. */
> +
> +static struct annotated * noinline alloc_buf_less (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index + SIZE_BUMP;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed
> + allocation. So, the object size info can only been obtained from
> + the call to malloc. */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +int main ()
> +{
> + struct annotated *p, *q;
> + p = alloc_buf_more (10);
> + q = alloc_buf_less (10);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0), -1);
> + expect(__builtin_dynamic_object_size(p->array, 1), p->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(p->array, 2), 0);
> + expect(__builtin_dynamic_object_size(p->array, 3), p->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(p, 0), -1);
> + expect(__builtin_dynamic_object_size(p, 1), -1);
> + expect(__builtin_dynamic_object_size(p, 2), 0);
> + expect(__builtin_dynamic_object_size(p, 3), 0);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(q->array, 0), -1);
> + expect(__builtin_dynamic_object_size(q->array, 1), q->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(q->array, 2), 0);
> + expect(__builtin_dynamic_object_size(q->array, 3), q->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(q, 0), -1);
> + expect(__builtin_dynamic_object_size(q, 1), -1);
> + expect(__builtin_dynamic_object_size(q, 2), 0);
> + expect(__builtin_dynamic_object_size(q, 3), 0);
> +
> + DONE ();
> +}
> diff --git a/gcc/tree-object-size.cc b/gcc/tree-object-size.cc
> index a62af0500563..cf7843c5684b 100644
> --- a/gcc/tree-object-size.cc
> +++ b/gcc/tree-object-size.cc
> @@ -585,6 +585,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> if (pt_var != TREE_OPERAND (ptr, 0))
> {
> tree var;
> + tree counted_by_ref = NULL_TREE;
>
> if (object_size_type & OST_SUBOBJECT)
> {
> @@ -600,11 +601,12 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> var = TREE_OPERAND (var, 0);
> if (var != pt_var && TREE_CODE (var) == ARRAY_REF)
> var = TREE_OPERAND (var, 0);
> - if (! TYPE_SIZE_UNIT (TREE_TYPE (var))
> + if (! component_ref_has_counted_by_p (var)
> + && ((! TYPE_SIZE_UNIT (TREE_TYPE (var))
> || ! tree_fits_uhwi_p (TYPE_SIZE_UNIT (TREE_TYPE (var)))
> || (pt_var_size && TREE_CODE (pt_var_size) == INTEGER_CST
> && tree_int_cst_lt (pt_var_size,
> - TYPE_SIZE_UNIT (TREE_TYPE (var)))))
> + TYPE_SIZE_UNIT (TREE_TYPE (var)))))))
> var = pt_var;
> else if (var != pt_var && TREE_CODE (pt_var) == MEM_REF)
> {
> @@ -612,6 +614,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* For &X->fld, compute object size if fld isn't a flexible array
> member. */
> bool is_flexible_array_mem_ref = false;
> +
> while (v && v != pt_var)
> switch (TREE_CODE (v))
> {
> @@ -660,6 +663,8 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* Now the ref is to an array type. */
> gcc_assert (TREE_CODE (TREE_TYPE (v)) == ARRAY_TYPE);
> is_flexible_array_mem_ref = array_ref_flexible_size_p (v);
> + counted_by_ref = component_ref_get_counted_by (v);
> +
> while (v != pt_var && TREE_CODE (v) == COMPONENT_REF)
> if (TREE_CODE (TREE_TYPE (TREE_OPERAND (v, 0)))
> != UNION_TYPE
> @@ -673,8 +678,11 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> == RECORD_TYPE)
> {
> /* compute object size only if v is not a
> - flexible array member. */
> - if (!is_flexible_array_mem_ref)
> + flexible array member or the flexible array member
> + has a known element count indicated by the user
> + through attribute counted_by. */
> + if (!is_flexible_array_mem_ref
> + || counted_by_ref)
> {
> v = NULL_TREE;
> break;
> @@ -707,9 +715,24 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>
> if (var != pt_var)
> {
> - var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> - if (!TREE_CONSTANT (var_size))
> - var_size = get_or_create_ssa_default_def (cfun, var_size);
> + if (!counted_by_ref)
> + {
> + var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> + if (!TREE_CONSTANT (var_size))
> + var_size = get_or_create_ssa_default_def (cfun, var_size);
> + }
> + else
> + {
> + gcc_assert (TREE_CODE (var) == COMPONENT_REF
> + && TREE_CODE (TREE_TYPE (var)) == ARRAY_TYPE);
> + tree element_size = TYPE_SIZE_UNIT (TREE_TYPE (TREE_TYPE (var)));
> + var_size
> + = size_binop (MULT_EXPR,
> + fold_convert (sizetype, counted_by_ref),
> + fold_convert (sizetype, element_size));
> + if (!todo)
> + todo = TODO_update_ssa_only_virtuals;
> + }
> if (!var_size)
> return false;
> }
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index fcd36ae0cd74..3b6ddcbdcbf8 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,32 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, return TRUE when
> + an counted_by attribute attached to the corresponding FIELD_DECL.
> + return FALSE otherwise. */
> +bool
> +component_ref_has_counted_by_p (tree array_ref)
> +{
> + if (TREE_CODE (array_ref) != COMPONENT_REF)
> + return false;
> +
> + if (TREE_CODE (TREE_TYPE (array_ref)) != ARRAY_TYPE)
> + return false;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> +
> + if (!RECORD_OR_UNION_TYPE_P (struct_type))
> + return false;
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return false;
> + return true;
> +}
> +
> /* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> @@ -12771,7 +12797,7 @@ get_named_field (tree fieldlist, const char *fieldname)
> fields inside it recursively. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> - fieldname)) != NULL_TREE)
> + fieldname)) != NULL_TREE)
> {
> named_field = tree_cons (NULL_TREE, field, named_field);
> break;
> @@ -12784,6 +12810,73 @@ get_named_field (tree fieldlist, const char *fieldname)
> return named_field;
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, get the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object.
> + For example, if:
> +
> + struct P {
> + int k;
> + int x[] __attribute__ ((counted_by (k)));
> + } *p;
> +
> + for the following reference:
> +
> + p->x[b]
> +
> + the object that represents its element count will be:
> +
> + p->k
> +
> + So, when component_ref_get_counted_by (p->x[b]) is called, p->k should be
> + returned.
> +*/
> +
> +tree
> +component_ref_get_counted_by (tree array_ref)
> +{
> + if (! component_ref_has_counted_by_p (array_ref))
> + return NULL_TREE;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> + gcc_assert (attr_counted_by);
> +
> + /* If there is an counted_by attribute attached to the field,
> + get the field that maps to the counted_by. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + tree counted_by_field = get_named_field (TYPE_FIELDS (struct_type),
> + fieldname);
> +
> + gcc_assert (counted_by_field);
> +
> + /* generate the tree node that represent the counted_by of this array
> + ref. This is a (possible nested) COMPONENT_REF to the counted_by_field
> + of the containing structure. */
> +
> + tree counted_by_ref = NULL_TREE;
> + tree object = struct_object;
> + do
> + {
> + tree field = TREE_VALUE (counted_by_field);
> +
> + counted_by_ref = build3 (COMPONENT_REF,
> + TREE_TYPE (field),
> + unshare_expr (object), field,
> + NULL_TREE);
> + object = counted_by_ref;
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + }
> + while (counted_by_field);
> + return counted_by_ref;
> +}
>
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4859becaa1e7..07eed7219835 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,11 +5619,21 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Give a component_ref that has an array type, return true when an
> + attribute counted_by attached to the corresponding FIELD_DECL. */
> +extern bool component_ref_has_counted_by_p (tree);
> +
> /* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> name is FIELDNAME, return NULL_TREE if such field is not found.
> searching nested anonymous structure/union recursively. */
> extern tree get_named_field (tree, const char *);
>
> +/* Give a component_ref that has an array type, return the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object. */
> +extern tree component_ref_get_counted_by (tree);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896]
2023-08-25 15:24 ` [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896] Qing Zhao
@ 2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:45 ` PING * 2: " Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-08 14:12 UTC (permalink / raw)
To: Joseph Myers, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Ping.
thanks.
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Use the counted_by attribute information in bound sanitizer.
>
> gcc/c-family/ChangeLog:
>
> PR C/108896
> * c-ubsan.cc (ubsan_instrument_bounds): Use counted_by attribute
> information.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/ubsan/flex-array-counted-by-bounds.c: New test.
> * gcc.dg/ubsan/flex-array-counted-by-bounds-2.c: New test.
> ---
> gcc/c-family/c-ubsan.cc | 16 +++++++
> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++++++++++
> .../ubsan/flex-array-counted-by-bounds.c | 46 +++++++++++++++++++
> 3 files changed, 89 insertions(+)
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
>
> diff --git a/gcc/c-family/c-ubsan.cc b/gcc/c-family/c-ubsan.cc
> index 51aa83a378d2..a99e8433069f 100644
> --- a/gcc/c-family/c-ubsan.cc
> +++ b/gcc/c-family/c-ubsan.cc
> @@ -362,6 +362,10 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> {
> tree type = TREE_TYPE (array);
> tree domain = TYPE_DOMAIN (type);
> + /* whether the array ref is a flexible array member with valid counted_by
> + attribute. */
> + bool fam_has_count_attr = false;
> + tree counted_by = NULL_TREE;
>
> if (domain == NULL_TREE)
> return NULL_TREE;
> @@ -375,6 +379,17 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> && COMPLETE_TYPE_P (type)
> && integer_zerop (TYPE_SIZE (type)))
> bound = build_int_cst (TREE_TYPE (TYPE_MIN_VALUE (domain)), -1);
> + /* If the array ref is to flexible array member field which has
> + counted_by attribute. We can use the information from the
> + attribute as the bound to instrument the reference. */
> + else if ((counted_by = component_ref_get_counted_by (array))
> + != NULL_TREE)
> + {
> + fam_has_count_attr = true;
> + bound = fold_build2 (MINUS_EXPR, TREE_TYPE (counted_by),
> + counted_by,
> + build_int_cst (TREE_TYPE (counted_by), 1));
> + }
> else
> return NULL_TREE;
> }
> @@ -387,6 +402,7 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> -fsanitize=bounds-strict. */
> tree base = get_base_address (array);
> if (!sanitize_flags_p (SANITIZE_BOUNDS_STRICT)
> + && !fam_has_count_attr
> && TREE_CODE (array) == COMPONENT_REF
> && base && (INDIRECT_REF_P (base) || TREE_CODE (base) == MEM_REF))
> {
> diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> new file mode 100644
> index 000000000000..77ec333509d0
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> @@ -0,0 +1,27 @@
> +/* test the attribute counted_by and its usage in
> + bounds sanitizer combined with VLA. */
> +/* { dg-do run } */
> +/* { dg-options "-fsanitize=bounds" } */
> +
> +#include <stdlib.h>
> +
> +void __attribute__((__noinline__)) setup_and_test_vla (int n, int m)
> +{
> + struct foo {
> + int n;
> + int p[][n] __attribute__((counted_by(n)));
> + } *f;
> +
> + f = (struct foo *) malloc (sizeof(struct foo) + m*sizeof(int[n]));
> + f->n = m;
> + f->p[m][n-1]=1;
> + return;
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup_and_test_vla (10, 11);
> + return 0;
> +}
> +
> +/* { dg-output "17:8: runtime error: index 11 out of bounds for type" } */
> diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
> new file mode 100644
> index 000000000000..81eaeb3f2681
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
> @@ -0,0 +1,46 @@
> +/* test the attribute counted_by and its usage in
> + bounds sanitizer. */
> +/* { dg-do run } */
> +/* { dg-options "-fsanitize=bounds" } */
> +
> +#include <stdlib.h>
> +
> +struct flex {
> + int b;
> + int c[];
> +} *array_flex;
> +
> +struct annotated {
> + int b;
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_annotated;
> +
> +void __attribute__((__noinline__)) setup (int normal_count, int annotated_count)
> +{
> + array_flex
> + = (struct flex *)malloc (sizeof (struct flex)
> + + normal_count * sizeof (int));
> + array_flex->b = normal_count;
> +
> + array_annotated
> + = (struct annotated *)malloc (sizeof (struct annotated)
> + + annotated_count * sizeof (int));
> + array_annotated->b = annotated_count;
> +
> + return;
> +}
> +
> +void __attribute__((__noinline__)) test (int normal_index, int annotated_index)
> +{
> + array_flex->c[normal_index] = 1;
> + array_annotated->c[annotated_index] = 2;
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup (10, 10);
> + test (10, 10);
> + return 0;
> +}
> +
> +/* { dg-output "36:21: runtime error: index 10 out of bounds for type" } */
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* PING * 2: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
` (4 preceding siblings ...)
2023-09-08 14:11 ` Qing Zhao
@ 2023-09-20 13:43 ` Qing Zhao
2023-10-05 20:08 ` Siddhesh Poyarekar
6 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-20 13:43 UTC (permalink / raw)
To: Joseph Myers, Richard Biener, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Hi,
I’d like to ping this patch set one more time!
Thanks.
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> This is the 3rd version of the patch, per our discussion based on the
> review comments for the 1st and 2nd version, the major changes in this
> version are:
>
> ***Against 1st version:
> 1. change the name "element_count" to "counted_by";
> 2. change the parameter for the attribute from a STRING to an
> Identifier;
> 3. Add logic and testing cases to handle anonymous structure/unions;
> 4. Clarify documentation to permit the situation when the allocation
> size is larger than what's specified by "counted_by", at the same time,
> it's user's error if allocation size is smaller than what's specified by
> "counted_by";
> 5. Add a complete testing case for using counted_by attribute in
> __builtin_dynamic_object_size when there is mismatch between the
> allocation size and the value of "counted_by", the expecting behavior
> for each case and the explanation on why in the comments.
>
> ***Against 2rd version:
> 1. Identify a tree node sharing issue and fixed it in the routine
> "component_ref_get_counted_ty" of tree.cc;
> 2. Update the documentation and testing cases with the clear usage
> of the fomula to compute the allocation size:
> MAX (sizeof (struct A), offsetof (struct A, array[0]) + counted_by * sizeof(element))
> (the algorithm used in tree-object-size.cc is correct).
>
> In this set of patches, the major functionality provided is:
>
> 1. a new attribute "counted_by";
> 2. use this new attribute in bound sanitizer;
> 3. use this new attribute in dynamic object size for subobject size;
>
> As discussed, I plan to add two more separate patches sets after this initial
> patch set is approved and committed.
>
> set 1. A new warning option and a new sanitizer option for the user error
> when the allocation size is smaller than the value of "counted_by".
> set 2. An improvement to __builtin_dynamic_object_size for whole-object
> size of the structure with FAM annaoted with counted_by.
>
> there are also some existing bugs in tree-object-size.cc identified
> during the study, and PRs were filed to record them. these bugs will
> be fixed seperately with individual patches:
>
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111030
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111040
>
> Bootstrapped and regression tested on both aarch64 and X86, no issue.
>
> Please see more details on the description of this work on:
>
> https://gcc.gnu.org/pipermail/gcc-patches/2023-May/619708.html
>
> and more discussions on
> https://gcc.gnu.org/pipermail/gcc-patches/2023-August/626376.html
>
> Okay for committing?
>
> thanks.
>
> Qing
>
> Qing Zhao (3):
> Provide counted_by attribute to flexible array member field (PR108896)
> Use the counted_by atribute info in builtin object size [PR108896]
> Use the counted_by attribute information in bound sanitizer[PR108896]
>
> gcc/c-family/c-attribs.cc | 54 ++++-
> gcc/c-family/c-common.cc | 13 ++
> gcc/c-family/c-common.h | 1 +
> gcc/c-family/c-ubsan.cc | 16 ++
> gcc/c/c-decl.cc | 79 +++++--
> gcc/doc/extend.texi | 77 +++++++
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++
> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++
> .../ubsan/flex-array-counted-by-bounds.c | 46 ++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 133 +++++++++++
> gcc/tree.h | 15 ++
> 14 files changed, 797 insertions(+), 25 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Ping * 2: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
2023-09-08 14:12 ` Qing Zhao
@ 2023-09-20 13:44 ` Qing Zhao
2023-10-05 18:51 ` Siddhesh Poyarekar
2 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-20 13:44 UTC (permalink / raw)
To: Joseph Myers, Richard Biener, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Hi,
I’d like to ping this patch set one more time.
Thanks
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Provide a new counted_by attribute to flexible array member field.
>
> 'counted_by (COUNT)'
> The 'counted_by' attribute may be attached to the flexible array
> member of a structure. It indicates that the number of the
> elements of the array is given by the field named "COUNT" in the
> same structure as the flexible array member. GCC uses this
> information to improve the results of the array bound sanitizer and
> the '__builtin_dynamic_object_size'.
>
> For instance, the following code:
>
> struct P {
> size_t count;
> char other;
> char array[] __attribute__ ((counted_by (count)));
> } *p;
>
> specifies that the 'array' is a flexible array member whose number
> of elements is given by the field 'count' in the same structure.
>
> The field that represents the number of the elements should have an
> integer type. An explicit 'counted_by' annotation defines a
> relationship between two objects, 'p->array' and 'p->count', that
> 'p->array' has _at least_ 'p->count' number of elements available.
> This relationship must hold even after any of these related objects
> are updated. It's the user's responsibility to make sure this
> relationship to be kept all the time. Otherwise the results of the
> array bound sanitizer and the '__builtin_dynamic_object_size' might
> be incorrect.
>
> For instance, in the following example, the allocated array has
> less elements than what's specified by the 'sbuf->count', this is
> an user error. As a result, out-of-bounds access to the array
> might not be detected.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + nelems * sizeof (char))));
> sbuf->count = nelems + SIZE_BUMP;
> /* This is invalid when the sbuf->array has less than sbuf->count
> elements. */
> }
>
> In the following example, the 2nd update to the field 'sbuf->count'
> of the above structure will permit out-of-bounds access to the
> array 'sbuf>array' as well.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + (nelems + SIZE_BUMP) * sizeof (char))));
> sbuf->count = nelems;
> /* This is valid when the sbuf->array has at least sbuf->count
> elements. */
> }
> void use_buf (int index)
> {
> sbuf->count = sbuf->count + SIZE_BUMP + 1;
> /* Now the value of sbuf->count is larger than the number
> of elements of sbuf->array. */
> sbuf->array[index] = 0;
> /* then the out-of-bound access to this array
> might not be detected. */
> }
>
> gcc/c-family/ChangeLog:
>
> PR C/108896
> * c-attribs.cc (handle_counted_by_attribute): New function.
> (attribute_takes_identifier_p): Add counted_by attribute to the list.
> * c-common.cc (c_flexible_array_member_type_p): ...To this.
> * c-common.h (c_flexible_array_member_type_p): New prototype.
>
> gcc/c/ChangeLog:
>
> PR C/108896
> * c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
> (add_flexible_array_elts_to_size): Use renamed function.
> (is_flexible_array_member_p): Use renamed function.
> (verify_counted_by_attribute): New function.
> (finish_struct): Use renamed function and verify counted_by
> attribute.
>
> gcc/ChangeLog:
>
> PR C/108896
> * doc/extend.texi: Document attribute counted_by.
> * tree.cc (get_named_field): New function.
> * tree.h (get_named_field): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by.c: New test.
> ---
> gcc/c-family/c-attribs.cc | 54 ++++++++++++-
> gcc/c-family/c-common.cc | 13 ++++
> gcc/c-family/c-common.h | 1 +
> gcc/c/c-decl.cc | 79 +++++++++++++++-----
> gcc/doc/extend.texi | 77 +++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
> gcc/tree.cc | 40 ++++++++++
> gcc/tree.h | 5 ++
> 8 files changed, 291 insertions(+), 18 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>
> diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
> index e2792ca6898b..65e4f6639109 100644
> --- a/gcc/c-family/c-attribs.cc
> +++ b/gcc/c-family/c-attribs.cc
> @@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute (tree *, tree, tree,
> int, bool *);
> static tree handle_strict_flex_array_attribute (tree *, tree, tree,
> int, bool *);
> +static tree handle_counted_by_attribute (tree *, tree, tree,
> + int, bool *);
> static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree, bool *);
> @@ -373,6 +375,8 @@ const struct attribute_spec c_common_attribute_table[] =
> handle_warn_if_not_aligned_attribute, NULL },
> { "strict_flex_array", 1, 1, true, false, false, false,
> handle_strict_flex_array_attribute, NULL },
> + { "counted_by", 1, 1, true, false, false, false,
> + handle_counted_by_attribute, NULL },
> { "weak", 0, 0, true, false, false, false,
> handle_weak_attribute, NULL },
> { "noplt", 0, 0, true, false, false, false,
> @@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
> else if (!strcmp ("mode", spec->name)
> || !strcmp ("format", spec->name)
> || !strcmp ("cleanup", spec->name)
> - || !strcmp ("access", spec->name))
> + || !strcmp ("access", spec->name)
> + || !strcmp ("counted_by", spec->name))
> return true;
> else
> return targetm.attribute_takes_identifier_p (attr_id);
> @@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node, tree name,
> return NULL_TREE;
> }
>
> +/* Handle a "counted_by" attribute; arguments as in
> + struct attribute_spec.handler. */
> +
> +static tree
> +handle_counted_by_attribute (tree *node, tree name,
> + tree args, int ARG_UNUSED (flags),
> + bool *no_add_attrs)
> +{
> + tree decl = *node;
> + tree argval = TREE_VALUE (args);
> +
> + /* This attribute only applies to field decls of a structure. */
> + if (TREE_CODE (decl) != FIELD_DECL)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for non-field"
> + " declaration %q+D", name, decl);
> + *no_add_attrs = true;
> + }
> + /* This attribute only applies to field with array type. */
> + else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non-array field",
> + name);
> + *no_add_attrs = true;
> + }
> + /* This attribute only applies to a C99 flexible array member type. */
> + else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non"
> + " flexible array member field",
> + name);
> + *no_add_attrs = true;
> + }
> + /* The argument should be an identifier. */
> + else if (TREE_CODE (argval) != IDENTIFIER_NODE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%<counted_by%> argument not an identifier");
> + *no_add_attrs = true;
> + }
> +
> + return NULL_TREE;
> +}
> +
> /* Handle a "weak" attribute; arguments as in
> struct attribute_spec.handler. */
>
> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
> index 9fbaeb437a12..a18937245c2a 100644
> --- a/gcc/c-family/c-common.cc
> +++ b/gcc/c-family/c-common.cc
> @@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
> (*debug_hooks->early_global_decl) (cnode->decl);
> }
>
> +/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> +bool
> +c_flexible_array_member_type_p (const_tree type)
> +{
> + if (TREE_CODE (type) == ARRAY_TYPE
> + && TYPE_SIZE (type) == NULL_TREE
> + && TYPE_DOMAIN (type) != NULL_TREE
> + && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> + return true;
> +
> + return false;
> +}
> +
> /* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based on the
> values of attribute strict_flex_array and the flag_strict_flex_arrays. */
> unsigned int
> diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
> index 78fc5248ba68..c29bb429062b 100644
> --- a/gcc/c-family/c-common.h
> +++ b/gcc/c-family/c-common.h
> @@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
> extern tree c_common_get_narrower (tree, int *);
> extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
> extern void c_common_finalize_early_debug (void);
> +extern bool c_flexible_array_member_type_p (const_tree);
> extern unsigned int c_strict_flex_array_level_of (tree);
> extern bool c_option_is_from_cpp_diagnostics (int);
>
> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
> index 1f9eb44dbaa2..e943b49b5230 100644
> --- a/gcc/c/c-decl.cc
> +++ b/gcc/c/c-decl.cc
> @@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator *decl,
> return decl;
> }
>
> -/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> -static bool
> -flexible_array_member_type_p (const_tree type)
> -{
> - if (TREE_CODE (type) == ARRAY_TYPE
> - && TYPE_SIZE (type) == NULL_TREE
> - && TYPE_DOMAIN (type) != NULL_TREE
> - && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> - return true;
> -
> - return false;
> -}
> -
> /* Determine whether TYPE is a one-element array type "[1]". */
> static bool
> one_element_array_type_p (const_tree type)
> @@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree init)
>
> elt = CONSTRUCTOR_ELTS (init)->last ().value;
> type = TREE_TYPE (elt);
> - if (flexible_array_member_type_p (type))
> + if (c_flexible_array_member_type_p (type))
> {
> complete_array_type (&type, elt, false);
> DECL_SIZE (decl)
> @@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
>
> bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
> bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
> - bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
> + bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE (x));
>
> unsigned int strict_flex_array_level = c_strict_flex_array_level_of (x);
>
> @@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
> return false;
> }
>
> +/* Verify the argument of the counted_by attribute of the flexible array
> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
> + FIELDLIST, Report error and remove this attribute when it's not. */
> +static void
> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
> +{
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return;
> +
> + /* If there is an counted_by attribute attached to the field,
> + verify it. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + /* Verify the argument of the attrbute is a valid field of the
> + containing structure. */
> +
> + tree counted_by_field = get_named_field (fieldlist, fieldname);
> +
> + /* Error when the field is not found in the containing structure. */
> + if (!counted_by_field)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " in the same structure, ignore it",
> + (get_attribute_name (attr_counted_by)));
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + else
> + /* Error when the field is not with an integer type. */
> + {
> + while (TREE_CHAIN (counted_by_field))
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + tree real_field = TREE_VALUE (counted_by_field);
> +
> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " with integer type, ignore it",
> + (get_attribute_name (attr_counted_by)));
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + }
> +
> + return;
> +}
>
> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
> LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
> @@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> DECL_PACKED (x) = 1;
>
> /* Detect flexible array member in an invalid context. */
> - if (flexible_array_member_type_p (TREE_TYPE (x)))
> + if (c_flexible_array_member_type_p (TREE_TYPE (x)))
> {
> if (TREE_CODE (t) == UNION_TYPE)
> {
> @@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> "members");
> TREE_TYPE (x) = error_mark_node;
> }
> + /* if there is a counted_by attribute attached to this field,
> + verify it. */
> + verify_counted_by_attribute (fieldlist, x);
> }
>
> if (pedantic && TREE_CODE (t) == RECORD_TYPE
> @@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> when x is an array and is the last field. */
> if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
> TYPE_INCLUDES_FLEXARRAY (t)
> - = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
> + = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
> /* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of x, t
> when x is an union or record and is the last field. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
> diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
> index 97eaacf8a7ec..ea6240646936 100644
> --- a/gcc/doc/extend.texi
> +++ b/gcc/doc/extend.texi
> @@ -7617,6 +7617,83 @@ When both the attribute and the option present at the same time, the level of
> the strictness for the specific trailing array field is determined by the
> attribute.
>
> +@cindex @code{counted_by} variable attribute
> +@item counted_by (@var{count})
> +The @code{counted_by} attribute may be attached to the flexible array
> +member of a structure. It indicates that the number of the elements of the
> +array is given by the field named "@var{count}" in the same structure as the
> +flexible array member. GCC uses this information to improve the results of
> +the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
> +
> +For instance, the following code:
> +
> +@smallexample
> +struct P @{
> + size_t count;
> + char other;
> + char array[] __attribute__ ((counted_by (count)));
> +@} *p;
> +@end smallexample
> +
> +@noindent
> +specifies that the @code{array} is a flexible array member whose number of
> +elements is given by the field @code{count} in the same structure.
> +
> +The field that represents the number of the elements should have an integer
> +type. An explicit @code{counted_by} annotation defines a relationship between
> +two objects, @code{p->array} and @code{p->count}, that @code{p->array} has
> +@emph{at least} @code{p->count} number of elements available. This relationship
> +must hold even after any of these related objects are updated. It's the user's
> +responsibility to make sure this relationship to be kept all the time.
> +Otherwise the results of the array bound sanitizer and the
> +@code{__builtin_dynamic_object_size} might be incorrect.
> +
> +For instance, in the following example, the allocated array has less elements
> +than what's specified by the @code{sbuf->count}, this is an user error. As a
> +result, out-of-bounds access to the array might not be detected.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + nelems * sizeof (char))));
> + sbuf->count = nelems + SIZE_BUMP;
> + /* This is invalid when the sbuf->array has less than sbuf->count
> + elements. */
> +@}
> +@end smallexample
> +
> +In the following example, the 2nd update to the field @code{sbuf->count} of
> +the above structure will permit out-of-bounds access to the array
> +@code{sbuf>array} as well.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + (nelems + SIZE_BUMP) * sizeof (char))));
> + sbuf->count = nelems;
> + /* This is valid when the sbuf->array has at least sbuf->count
> + elements. */
> +@}
> +void use_buf (int index)
> +@{
> + sbuf->count = sbuf->count + SIZE_BUMP + 1;
> + /* Now the value of sbuf->count is larger than the number
> + of elements of sbuf->array. */
> + sbuf->array[index] = 0;
> + /* then the out-of-bound access to this array
> + might not be detected. */
> +@}
> +@end smallexample
> +
> +
> @cindex @code{alloc_size} variable attribute
> @item alloc_size (@var{position})
> @itemx alloc_size (@var{position-1}, @var{position-2})
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> new file mode 100644
> index 000000000000..f8ce9776bf86
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> @@ -0,0 +1,40 @@
> +/* testing the correct usage of attribute counted_by. */
> +/* { dg-do compile } */
> +/* { dg-options "-O2" } */
> +
> +#include <wchar.h>
> +
> +int size;
> +int x __attribute ((counted_by (size))); /* { dg-error "attribute may not be specified for non-field declaration" } */
> +
> +struct trailing {
> + int count;
> + int field __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non-array field" } */
> +};
> +
> +struct trailing_1 {
> + int count;
> + int array_1[0] __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non flexible array member field" } */
> +};
> +
> +int count;
> +struct trailing_array_2 {
> + int count;
> + int array_2[] __attribute ((counted_by ("count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_3 {
> + int other;
> + int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_4 {
> + int other;
> + int array_4[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration in the same structure, ignore it" } */
> +};
> +
> +int count;
> +struct trailing_array_5 {
> + float count;
> + int array_5[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration with integer type, ignore it" } */
> +};
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index 420857b110c4..fcd36ae0cd74 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> + with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> + whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> + return NULL_TREE if such field is not found. Normally this list is of
> + length one, but if the field is embedded with (nested) anonymous structures
> + or unions, this list steps down the chain to the field. */
> +tree
> +get_named_field (tree fieldlist, const char *fieldname)
> +{
> + tree named_field = NULL_TREE;
> + for (tree field = fieldlist; field; field = DECL_CHAIN (field))
> + {
> + if (TREE_CODE (field) != FIELD_DECL)
> + continue;
> + if (DECL_NAME (field) != NULL)
> + if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + /* if the field is an anonymous struct/union, we will check the nested
> + fields inside it recursively. */
> + else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> + if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> + fieldname)) != NULL_TREE)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + else
> + continue;
> + }
> + return named_field;
> +}
> +
> +
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4c04245e2b1b..4859becaa1e7 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> + name is FIELDNAME, return NULL_TREE if such field is not found.
> + searching nested anonymous structure/union recursively. */
> +extern tree get_named_field (tree, const char *);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* PING *2: [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896]
2023-08-25 15:24 ` [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896] Qing Zhao
2023-09-08 14:12 ` Qing Zhao
@ 2023-09-20 13:44 ` Qing Zhao
2023-10-05 20:01 ` Siddhesh Poyarekar
2 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-20 13:44 UTC (permalink / raw)
To: Joseph Myers, richard.guenther, jakub, gcc-patches
Cc: keescook, siddhesh, uecker, isanbard
Hi,
I’d like to ping this patch set one more time.
Thanks
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Use the counted_by atribute info in builtin object size to compute the
> subobject size for flexible array members.
>
> gcc/ChangeLog:
>
> PR C/108896
> * tree-object-size.cc (addr_object_size): Use the counted_by
> attribute info.
> * tree.cc (component_ref_has_counted_by_p): New function.
> (component_ref_get_counted_by): New function.
> * tree.h (component_ref_has_counted_by_p): New prototype.
> (component_ref_get_counted_by): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by-2.c: New test.
> * gcc.dg/flex-array-counted-by-3.c: New test.
> ---
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 95 +++++++-
> gcc/tree.h | 10 +
> 5 files changed, 418 insertions(+), 8 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> new file mode 100644
> index 000000000000..ec580c1f1f01
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> @@ -0,0 +1,74 @@
> +/* test the attribute counted_by and its usage in
> + * __builtin_dynamic_object_size. */
> +/* { dg-do run } */
> +/* { dg-options "-O2" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
> +
> +struct flex {
> + int b;
> + int c[];
> +} *array_flex;
> +
> +struct annotated {
> + int b;
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_annotated;
> +
> +struct nested_annotated {
> + struct {
> + union {
> + int b;
> + float f;
> + };
> + int n;
> + };
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_nested_annotated;
> +
> +void __attribute__((__noinline__)) setup (int normal_count, int attr_count)
> +{
> + array_flex
> + = (struct flex *)malloc (sizeof (struct flex)
> + + normal_count * sizeof (int));
> + array_flex->b = normal_count;
> +
> + array_annotated
> + = (struct annotated *)malloc (sizeof (struct annotated)
> + + attr_count * sizeof (int));
> + array_annotated->b = attr_count;
> +
> + array_nested_annotated
> + = (struct nested_annotated *)malloc (sizeof (struct nested_annotated)
> + + attr_count * sizeof (int));
> + array_nested_annotated->b = attr_count;
> +
> + return;
> +}
> +
> +void __attribute__((__noinline__)) test ()
> +{
> + expect(__builtin_dynamic_object_size(array_flex->c, 1), -1);
> + expect(__builtin_dynamic_object_size(array_annotated->c, 1),
> + array_annotated->b * sizeof (int));
> + expect(__builtin_dynamic_object_size(array_nested_annotated->c, 1),
> + array_nested_annotated->b * sizeof (int));
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup (10,10);
> + test ();
> + DONE ();
> +}
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> new file mode 100644
> index 000000000000..a0c3cb88ec71
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> @@ -0,0 +1,210 @@
> +/* test the attribute counted_by and its usage in
> +__builtin_dynamic_object_size: what's the correct behavior when the
> +allocation size mismatched with the value of counted_by attribute? */
> +/* { dg-do run } */
> +/* { dg-options "-O -fstrict-flex-arrays=3" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +struct annotated {
> + size_t foo;
> + char others;
> + char array[] __attribute__((counted_by (foo)));
> +};
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
> +
> +#define noinline __attribute__((__noinline__))
> +#define SIZE_BUMP 10
> +#define MAX(a, b) ((a) > (b) ? (a) : (b))
> +#define MIN(a, b) ((a) < (b) ? (a) : (b))
> +
> +/* In general, Due to type casting, the type for the pointee of a pointer
> + does not say anything about the object it points to,
> + So, __builtin_object_size can not directly use the type of the pointee
> + to decide the size of the object the pointer points to.
> +
> + there are only two reliable ways:
> + A. observed allocations (call to the allocation functions in the routine)
> + B. observed accesses (read or write access to the location of the
> + pointer points to)
> +
> + that provide information about the type/existence of an object at
> + the corresponding address.
> +
> + for A, we use the "alloc_size" attribute for the corresponding allocation
> + functions to determine the object size;
> +
> + For B, we use the SIZE info of the TYPE attached to the corresponding access.
> + (We treat counted_by attribute as a complement to the SIZE info of the TYPE
> + for FMA)
> +
> + The only other way in C which ensures that a pointer actually points
> + to an object of the correct type is 'static':
> +
> + void foo(struct P *p[static 1]);
> +
> + See https://gcc.gnu.org/pipermail/gcc-patches/2023-July/624814.html
> + for more details. */
> +
> +/* in the following function, malloc allocated more space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases? */
> +
> +static struct annotated * noinline alloc_buf_more (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index + SIZE_BUMP) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed allocation.
> + So, the object size info can only been obtained from the call to malloc.
> + for both MAXIMUM and MINIMUM: A = (index + SIZE_BUMP) * sizeof (char) */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +/* in the following function, malloc allocated less space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases?
> + NOTE: this is an user error, GCC should issue warnings for such case.
> + this is a seperate issue we should address later. */
> +
> +static struct annotated * noinline alloc_buf_less (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index + SIZE_BUMP;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed
> + allocation. So, the object size info can only been obtained from
> + the call to malloc. */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +int main ()
> +{
> + struct annotated *p, *q;
> + p = alloc_buf_more (10);
> + q = alloc_buf_less (10);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0), -1);
> + expect(__builtin_dynamic_object_size(p->array, 1), p->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(p->array, 2), 0);
> + expect(__builtin_dynamic_object_size(p->array, 3), p->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(p, 0), -1);
> + expect(__builtin_dynamic_object_size(p, 1), -1);
> + expect(__builtin_dynamic_object_size(p, 2), 0);
> + expect(__builtin_dynamic_object_size(p, 3), 0);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(q->array, 0), -1);
> + expect(__builtin_dynamic_object_size(q->array, 1), q->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(q->array, 2), 0);
> + expect(__builtin_dynamic_object_size(q->array, 3), q->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(q, 0), -1);
> + expect(__builtin_dynamic_object_size(q, 1), -1);
> + expect(__builtin_dynamic_object_size(q, 2), 0);
> + expect(__builtin_dynamic_object_size(q, 3), 0);
> +
> + DONE ();
> +}
> diff --git a/gcc/tree-object-size.cc b/gcc/tree-object-size.cc
> index a62af0500563..cf7843c5684b 100644
> --- a/gcc/tree-object-size.cc
> +++ b/gcc/tree-object-size.cc
> @@ -585,6 +585,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> if (pt_var != TREE_OPERAND (ptr, 0))
> {
> tree var;
> + tree counted_by_ref = NULL_TREE;
>
> if (object_size_type & OST_SUBOBJECT)
> {
> @@ -600,11 +601,12 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> var = TREE_OPERAND (var, 0);
> if (var != pt_var && TREE_CODE (var) == ARRAY_REF)
> var = TREE_OPERAND (var, 0);
> - if (! TYPE_SIZE_UNIT (TREE_TYPE (var))
> + if (! component_ref_has_counted_by_p (var)
> + && ((! TYPE_SIZE_UNIT (TREE_TYPE (var))
> || ! tree_fits_uhwi_p (TYPE_SIZE_UNIT (TREE_TYPE (var)))
> || (pt_var_size && TREE_CODE (pt_var_size) == INTEGER_CST
> && tree_int_cst_lt (pt_var_size,
> - TYPE_SIZE_UNIT (TREE_TYPE (var)))))
> + TYPE_SIZE_UNIT (TREE_TYPE (var)))))))
> var = pt_var;
> else if (var != pt_var && TREE_CODE (pt_var) == MEM_REF)
> {
> @@ -612,6 +614,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* For &X->fld, compute object size if fld isn't a flexible array
> member. */
> bool is_flexible_array_mem_ref = false;
> +
> while (v && v != pt_var)
> switch (TREE_CODE (v))
> {
> @@ -660,6 +663,8 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* Now the ref is to an array type. */
> gcc_assert (TREE_CODE (TREE_TYPE (v)) == ARRAY_TYPE);
> is_flexible_array_mem_ref = array_ref_flexible_size_p (v);
> + counted_by_ref = component_ref_get_counted_by (v);
> +
> while (v != pt_var && TREE_CODE (v) == COMPONENT_REF)
> if (TREE_CODE (TREE_TYPE (TREE_OPERAND (v, 0)))
> != UNION_TYPE
> @@ -673,8 +678,11 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> == RECORD_TYPE)
> {
> /* compute object size only if v is not a
> - flexible array member. */
> - if (!is_flexible_array_mem_ref)
> + flexible array member or the flexible array member
> + has a known element count indicated by the user
> + through attribute counted_by. */
> + if (!is_flexible_array_mem_ref
> + || counted_by_ref)
> {
> v = NULL_TREE;
> break;
> @@ -707,9 +715,24 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>
> if (var != pt_var)
> {
> - var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> - if (!TREE_CONSTANT (var_size))
> - var_size = get_or_create_ssa_default_def (cfun, var_size);
> + if (!counted_by_ref)
> + {
> + var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> + if (!TREE_CONSTANT (var_size))
> + var_size = get_or_create_ssa_default_def (cfun, var_size);
> + }
> + else
> + {
> + gcc_assert (TREE_CODE (var) == COMPONENT_REF
> + && TREE_CODE (TREE_TYPE (var)) == ARRAY_TYPE);
> + tree element_size = TYPE_SIZE_UNIT (TREE_TYPE (TREE_TYPE (var)));
> + var_size
> + = size_binop (MULT_EXPR,
> + fold_convert (sizetype, counted_by_ref),
> + fold_convert (sizetype, element_size));
> + if (!todo)
> + todo = TODO_update_ssa_only_virtuals;
> + }
> if (!var_size)
> return false;
> }
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index fcd36ae0cd74..3b6ddcbdcbf8 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,32 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, return TRUE when
> + an counted_by attribute attached to the corresponding FIELD_DECL.
> + return FALSE otherwise. */
> +bool
> +component_ref_has_counted_by_p (tree array_ref)
> +{
> + if (TREE_CODE (array_ref) != COMPONENT_REF)
> + return false;
> +
> + if (TREE_CODE (TREE_TYPE (array_ref)) != ARRAY_TYPE)
> + return false;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> +
> + if (!RECORD_OR_UNION_TYPE_P (struct_type))
> + return false;
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return false;
> + return true;
> +}
> +
> /* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> @@ -12771,7 +12797,7 @@ get_named_field (tree fieldlist, const char *fieldname)
> fields inside it recursively. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> - fieldname)) != NULL_TREE)
> + fieldname)) != NULL_TREE)
> {
> named_field = tree_cons (NULL_TREE, field, named_field);
> break;
> @@ -12784,6 +12810,73 @@ get_named_field (tree fieldlist, const char *fieldname)
> return named_field;
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, get the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object.
> + For example, if:
> +
> + struct P {
> + int k;
> + int x[] __attribute__ ((counted_by (k)));
> + } *p;
> +
> + for the following reference:
> +
> + p->x[b]
> +
> + the object that represents its element count will be:
> +
> + p->k
> +
> + So, when component_ref_get_counted_by (p->x[b]) is called, p->k should be
> + returned.
> +*/
> +
> +tree
> +component_ref_get_counted_by (tree array_ref)
> +{
> + if (! component_ref_has_counted_by_p (array_ref))
> + return NULL_TREE;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> + gcc_assert (attr_counted_by);
> +
> + /* If there is an counted_by attribute attached to the field,
> + get the field that maps to the counted_by. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + tree counted_by_field = get_named_field (TYPE_FIELDS (struct_type),
> + fieldname);
> +
> + gcc_assert (counted_by_field);
> +
> + /* generate the tree node that represent the counted_by of this array
> + ref. This is a (possible nested) COMPONENT_REF to the counted_by_field
> + of the containing structure. */
> +
> + tree counted_by_ref = NULL_TREE;
> + tree object = struct_object;
> + do
> + {
> + tree field = TREE_VALUE (counted_by_field);
> +
> + counted_by_ref = build3 (COMPONENT_REF,
> + TREE_TYPE (field),
> + unshare_expr (object), field,
> + NULL_TREE);
> + object = counted_by_ref;
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + }
> + while (counted_by_field);
> + return counted_by_ref;
> +}
>
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4859becaa1e7..07eed7219835 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,11 +5619,21 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Give a component_ref that has an array type, return true when an
> + attribute counted_by attached to the corresponding FIELD_DECL. */
> +extern bool component_ref_has_counted_by_p (tree);
> +
> /* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> name is FIELDNAME, return NULL_TREE if such field is not found.
> searching nested anonymous structure/union recursively. */
> extern tree get_named_field (tree, const char *);
>
> +/* Give a component_ref that has an array type, return the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object. */
> +extern tree component_ref_get_counted_by (tree);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* PING * 2: [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896]
2023-08-25 15:24 ` [V3][PATCH 3/3] Use the counted_by attribute information in bound sanitizer[PR108896] Qing Zhao
2023-09-08 14:12 ` Qing Zhao
@ 2023-09-20 13:45 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-09-20 13:45 UTC (permalink / raw)
To: Joseph Myers, Richard Biener, jakub Jelinek, gcc Patches
Cc: keescook, siddhesh, uecker, isanbard
Hi,
I’d like to ping this patch set one more time.
Thanks
Qing
> On Aug 25, 2023, at 11:24 AM, Qing Zhao <qing.zhao@oracle.com> wrote:
>
> Use the counted_by attribute information in bound sanitizer.
>
> gcc/c-family/ChangeLog:
>
> PR C/108896
> * c-ubsan.cc (ubsan_instrument_bounds): Use counted_by attribute
> information.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/ubsan/flex-array-counted-by-bounds.c: New test.
> * gcc.dg/ubsan/flex-array-counted-by-bounds-2.c: New test.
> ---
> gcc/c-family/c-ubsan.cc | 16 +++++++
> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++++++++++
> .../ubsan/flex-array-counted-by-bounds.c | 46 +++++++++++++++++++
> 3 files changed, 89 insertions(+)
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
>
> diff --git a/gcc/c-family/c-ubsan.cc b/gcc/c-family/c-ubsan.cc
> index 51aa83a378d2..a99e8433069f 100644
> --- a/gcc/c-family/c-ubsan.cc
> +++ b/gcc/c-family/c-ubsan.cc
> @@ -362,6 +362,10 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> {
> tree type = TREE_TYPE (array);
> tree domain = TYPE_DOMAIN (type);
> + /* whether the array ref is a flexible array member with valid counted_by
> + attribute. */
> + bool fam_has_count_attr = false;
> + tree counted_by = NULL_TREE;
>
> if (domain == NULL_TREE)
> return NULL_TREE;
> @@ -375,6 +379,17 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> && COMPLETE_TYPE_P (type)
> && integer_zerop (TYPE_SIZE (type)))
> bound = build_int_cst (TREE_TYPE (TYPE_MIN_VALUE (domain)), -1);
> + /* If the array ref is to flexible array member field which has
> + counted_by attribute. We can use the information from the
> + attribute as the bound to instrument the reference. */
> + else if ((counted_by = component_ref_get_counted_by (array))
> + != NULL_TREE)
> + {
> + fam_has_count_attr = true;
> + bound = fold_build2 (MINUS_EXPR, TREE_TYPE (counted_by),
> + counted_by,
> + build_int_cst (TREE_TYPE (counted_by), 1));
> + }
> else
> return NULL_TREE;
> }
> @@ -387,6 +402,7 @@ ubsan_instrument_bounds (location_t loc, tree array, tree *index,
> -fsanitize=bounds-strict. */
> tree base = get_base_address (array);
> if (!sanitize_flags_p (SANITIZE_BOUNDS_STRICT)
> + && !fam_has_count_attr
> && TREE_CODE (array) == COMPONENT_REF
> && base && (INDIRECT_REF_P (base) || TREE_CODE (base) == MEM_REF))
> {
> diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> new file mode 100644
> index 000000000000..77ec333509d0
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> @@ -0,0 +1,27 @@
> +/* test the attribute counted_by and its usage in
> + bounds sanitizer combined with VLA. */
> +/* { dg-do run } */
> +/* { dg-options "-fsanitize=bounds" } */
> +
> +#include <stdlib.h>
> +
> +void __attribute__((__noinline__)) setup_and_test_vla (int n, int m)
> +{
> + struct foo {
> + int n;
> + int p[][n] __attribute__((counted_by(n)));
> + } *f;
> +
> + f = (struct foo *) malloc (sizeof(struct foo) + m*sizeof(int[n]));
> + f->n = m;
> + f->p[m][n-1]=1;
> + return;
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup_and_test_vla (10, 11);
> + return 0;
> +}
> +
> +/* { dg-output "17:8: runtime error: index 11 out of bounds for type" } */
> diff --git a/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
> new file mode 100644
> index 000000000000..81eaeb3f2681
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
> @@ -0,0 +1,46 @@
> +/* test the attribute counted_by and its usage in
> + bounds sanitizer. */
> +/* { dg-do run } */
> +/* { dg-options "-fsanitize=bounds" } */
> +
> +#include <stdlib.h>
> +
> +struct flex {
> + int b;
> + int c[];
> +} *array_flex;
> +
> +struct annotated {
> + int b;
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_annotated;
> +
> +void __attribute__((__noinline__)) setup (int normal_count, int annotated_count)
> +{
> + array_flex
> + = (struct flex *)malloc (sizeof (struct flex)
> + + normal_count * sizeof (int));
> + array_flex->b = normal_count;
> +
> + array_annotated
> + = (struct annotated *)malloc (sizeof (struct annotated)
> + + annotated_count * sizeof (int));
> + array_annotated->b = annotated_count;
> +
> + return;
> +}
> +
> +void __attribute__((__noinline__)) test (int normal_index, int annotated_index)
> +{
> + array_flex->c[normal_index] = 1;
> + array_annotated->c[annotated_index] = 2;
> +}
> +
> +int main(int argc, char *argv[])
> +{
> + setup (10, 10);
> + test (10, 10);
> + return 0;
> +}
> +
> +/* { dg-output "36:21: runtime error: index 10 out of bounds for type" } */
> --
> 2.31.1
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-08-25 15:24 ` [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896) Qing Zhao
2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:44 ` Ping * 2: " Qing Zhao
@ 2023-10-05 18:51 ` Siddhesh Poyarekar
2023-10-05 19:31 ` Siddhesh Poyarekar
2023-10-18 14:41 ` Qing Zhao
2 siblings, 2 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-05 18:51 UTC (permalink / raw)
To: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, uecker, isanbard
On 2023-08-25 11:24, Qing Zhao wrote:
> Provide a new counted_by attribute to flexible array member field.
The obligatory "I can't ack the patch but here's a review" disclaimer :)
>
> 'counted_by (COUNT)'
> The 'counted_by' attribute may be attached to the flexible array
> member of a structure. It indicates that the number of the
> elements of the array is given by the field named "COUNT" in the
> same structure as the flexible array member. GCC uses this
> information to improve the results of the array bound sanitizer and
> the '__builtin_dynamic_object_size'.
>
> For instance, the following code:
>
> struct P {
> size_t count;
> char other;
> char array[] __attribute__ ((counted_by (count)));
> } *p;
>
> specifies that the 'array' is a flexible array member whose number
> of elements is given by the field 'count' in the same structure.
>
> The field that represents the number of the elements should have an
> integer type. An explicit 'counted_by' annotation defines a
> relationship between two objects, 'p->array' and 'p->count', that
> 'p->array' has _at least_ 'p->count' number of elements available.
> This relationship must hold even after any of these related objects
> are updated. It's the user's responsibility to make sure this
> relationship to be kept all the time. Otherwise the results of the
> array bound sanitizer and the '__builtin_dynamic_object_size' might
> be incorrect.
>
> For instance, in the following example, the allocated array has
> less elements than what's specified by the 'sbuf->count', this is
> an user error. As a result, out-of-bounds access to the array
> might not be detected.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + nelems * sizeof (char))));
> sbuf->count = nelems + SIZE_BUMP;
> /* This is invalid when the sbuf->array has less than sbuf->count
> elements. */
> }
>
> In the following example, the 2nd update to the field 'sbuf->count'
> of the above structure will permit out-of-bounds access to the
> array 'sbuf>array' as well.
>
> #define SIZE_BUMP 10
> struct P *sbuf;
> void alloc_buf (size_t nelems)
> {
> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> (offsetof (struct P, array[0])
> + (nelems + SIZE_BUMP) * sizeof (char))));
> sbuf->count = nelems;
> /* This is valid when the sbuf->array has at least sbuf->count
> elements. */
> }
> void use_buf (int index)
> {
> sbuf->count = sbuf->count + SIZE_BUMP + 1;
> /* Now the value of sbuf->count is larger than the number
> of elements of sbuf->array. */
> sbuf->array[index] = 0;
> /* then the out-of-bound access to this array
> might not be detected. */
> }
>
> gcc/c-family/ChangeLog:
>
> PR C/108896
> * c-attribs.cc (handle_counted_by_attribute): New function.
> (attribute_takes_identifier_p): Add counted_by attribute to the list.
> * c-common.cc (c_flexible_array_member_type_p): ...To this.
> * c-common.h (c_flexible_array_member_type_p): New prototype.
>
> gcc/c/ChangeLog:
>
> PR C/108896
> * c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
> (add_flexible_array_elts_to_size): Use renamed function.
> (is_flexible_array_member_p): Use renamed function.
> (verify_counted_by_attribute): New function.
> (finish_struct): Use renamed function and verify counted_by
> attribute.
>
> gcc/ChangeLog:
>
> PR C/108896
> * doc/extend.texi: Document attribute counted_by.
> * tree.cc (get_named_field): New function.
> * tree.h (get_named_field): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by.c: New test.
> ---
> gcc/c-family/c-attribs.cc | 54 ++++++++++++-
> gcc/c-family/c-common.cc | 13 ++++
> gcc/c-family/c-common.h | 1 +
> gcc/c/c-decl.cc | 79 +++++++++++++++-----
> gcc/doc/extend.texi | 77 +++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
> gcc/tree.cc | 40 ++++++++++
> gcc/tree.h | 5 ++
> 8 files changed, 291 insertions(+), 18 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>
> diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
> index e2792ca6898b..65e4f6639109 100644
> --- a/gcc/c-family/c-attribs.cc
> +++ b/gcc/c-family/c-attribs.cc
> @@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute (tree *, tree, tree,
> int, bool *);
> static tree handle_strict_flex_array_attribute (tree *, tree, tree,
> int, bool *);
> +static tree handle_counted_by_attribute (tree *, tree, tree,
> + int, bool *);
> static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
> static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree, bool *);
> @@ -373,6 +375,8 @@ const struct attribute_spec c_common_attribute_table[] =
> handle_warn_if_not_aligned_attribute, NULL },
> { "strict_flex_array", 1, 1, true, false, false, false,
> handle_strict_flex_array_attribute, NULL },
> + { "counted_by", 1, 1, true, false, false, false,
> + handle_counted_by_attribute, NULL },
> { "weak", 0, 0, true, false, false, false,
> handle_weak_attribute, NULL },
> { "noplt", 0, 0, true, false, false, false,
> @@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
> else if (!strcmp ("mode", spec->name)
> || !strcmp ("format", spec->name)
> || !strcmp ("cleanup", spec->name)
> - || !strcmp ("access", spec->name))
> + || !strcmp ("access", spec->name)
> + || !strcmp ("counted_by", spec->name))
> return true;
> else
> return targetm.attribute_takes_identifier_p (attr_id);
> @@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node, tree name,
> return NULL_TREE;
> }
>
> +/* Handle a "counted_by" attribute; arguments as in
> + struct attribute_spec.handler. */
> +
> +static tree
> +handle_counted_by_attribute (tree *node, tree name,
> + tree args, int ARG_UNUSED (flags),
> + bool *no_add_attrs)
> +{
> + tree decl = *node;
> + tree argval = TREE_VALUE (args);
> +
> + /* This attribute only applies to field decls of a structure. */
> + if (TREE_CODE (decl) != FIELD_DECL)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for non-field"
> + " declaration %q+D", name, decl);
> + *no_add_attrs = true;
> + }
Applies only to struct fields. OK.
> + /* This attribute only applies to field with array type. */
> + else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non-array field",
> + name);
> + *no_add_attrs = true;
> + }
The struct field should also be an array. OK.
> + /* This attribute only applies to a C99 flexible array member type. */
> + else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%qE attribute may not be specified for a non"
> + " flexible array member field",
> + name);
> + *no_add_attrs = true;
> + }
Additionally, the field should be a *flex* array. OK. Could this be
reworded to:
%qE attribute only applies to C99 flexible array members
That would make it clear that the GNU extension flex arrays (i.e. any
last array member of a struct) don't support this attribute.
> + /* The argument should be an identifier. */
> + else if (TREE_CODE (argval) != IDENTIFIER_NODE)
> + {
> + error_at (DECL_SOURCE_LOCATION (decl),
> + "%<counted_by%> argument not an identifier");
> + *no_add_attrs = true;
> + }
Argument should be an identifier, and the check for validity of the
identifier comes later in finish_struct. OK.
> +
> + return NULL_TREE;
> +}
> +
> /* Handle a "weak" attribute; arguments as in
> struct attribute_spec.handler. */
>
> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
> index 9fbaeb437a12..a18937245c2a 100644
> --- a/gcc/c-family/c-common.cc
> +++ b/gcc/c-family/c-common.cc
> @@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
> (*debug_hooks->early_global_decl) (cnode->decl);
> }
>
> +/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> +bool
> +c_flexible_array_member_type_p (const_tree type)
> +{
> + if (TREE_CODE (type) == ARRAY_TYPE
> + && TYPE_SIZE (type) == NULL_TREE
> + && TYPE_DOMAIN (type) != NULL_TREE
> + && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> + return true;
> +
> + return false;
> +}
> +
Hoist flexible_array_member_type_p to use more widely. OK.
> /* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based on the
> values of attribute strict_flex_array and the flag_strict_flex_arrays. */
> unsigned int
> diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
> index 78fc5248ba68..c29bb429062b 100644
> --- a/gcc/c-family/c-common.h
> +++ b/gcc/c-family/c-common.h
> @@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
> extern tree c_common_get_narrower (tree, int *);
> extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
> extern void c_common_finalize_early_debug (void);
> +extern bool c_flexible_array_member_type_p (const_tree);
> extern unsigned int c_strict_flex_array_level_of (tree);
> extern bool c_option_is_from_cpp_diagnostics (int);
>
> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
> index 1f9eb44dbaa2..e943b49b5230 100644
> --- a/gcc/c/c-decl.cc
> +++ b/gcc/c/c-decl.cc
> @@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator *decl,
> return decl;
> }
>
> -/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
> -static bool
> -flexible_array_member_type_p (const_tree type)
> -{
> - if (TREE_CODE (type) == ARRAY_TYPE
> - && TYPE_SIZE (type) == NULL_TREE
> - && TYPE_DOMAIN (type) != NULL_TREE
> - && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
> - return true;
> -
> - return false;
> -}
> -
> /* Determine whether TYPE is a one-element array type "[1]". */
> static bool
> one_element_array_type_p (const_tree type)
> @@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree init)
>
> elt = CONSTRUCTOR_ELTS (init)->last ().value;
> type = TREE_TYPE (elt);
> - if (flexible_array_member_type_p (type))
> + if (c_flexible_array_member_type_p (type))
> {
> complete_array_type (&type, elt, false);
> DECL_SIZE (decl)
> @@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
>
> bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
> bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
> - bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
> + bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE (x));
>
> unsigned int strict_flex_array_level = c_strict_flex_array_level_of (x);
>
Simple refactoring. OK.
> @@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
> return false;
> }
>
> +/* Verify the argument of the counted_by attribute of the flexible array
Verify *that* the argument...
> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
> + FIELDLIST, Report error and remove this attribute when it's not. */
> +static void
> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
> +{
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return;
> +
> + /* If there is an counted_by attribute attached to the field,
> + verify it. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + /* Verify the argument of the attrbute is a valid field of the
s/attrbute/attribute/
> + containing structure. */
> +
> + tree counted_by_field = get_named_field (fieldlist, fieldname);
> +
> + /* Error when the field is not found in the containing structure. */
> + if (!counted_by_field)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " in the same structure, ignore it",
> + (get_attribute_name (attr_counted_by)));
Probably someone with English as a first language would make a better
suggestion, but how about:
Argument specified in %qE attribute is not a field declaration in the
same structure, ignoring it.
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + else
> + /* Error when the field is not with an integer type. */
Suggest: Flag an error when the field is not of an integer type.
> + {
> + while (TREE_CHAIN (counted_by_field))
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + tree real_field = TREE_VALUE (counted_by_field);
> +
> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
> + {
> + error_at (DECL_SOURCE_LOCATION (field_decl),
> + "%qE attribute argument not a field declaration"
> + " with integer type, ignore it",
> + (get_attribute_name (attr_counted_by)));
Suggest:
Argument specified in %qE attribute is not of an integer type,
ignoring it.
> +
> + DECL_ATTRIBUTES (field_decl)
> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
> + }
> + }
> +
> + return;
> +}
>
> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
> LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
> @@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> DECL_PACKED (x) = 1;
>
> /* Detect flexible array member in an invalid context. */
> - if (flexible_array_member_type_p (TREE_TYPE (x)))
> + if (c_flexible_array_member_type_p (TREE_TYPE (x)))
> {
> if (TREE_CODE (t) == UNION_TYPE)
> {
> @@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> "members");
> TREE_TYPE (x) = error_mark_node;
> }
> + /* if there is a counted_by attribute attached to this field,
> + verify it. */
> + verify_counted_by_attribute (fieldlist, x);
> }
>
> if (pedantic && TREE_CODE (t) == RECORD_TYPE
> @@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
> when x is an array and is the last field. */
> if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
> TYPE_INCLUDES_FLEXARRAY (t)
> - = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
> + = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
> /* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of x, t
> when x is an union or record and is the last field. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
> diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
> index 97eaacf8a7ec..ea6240646936 100644
> --- a/gcc/doc/extend.texi
> +++ b/gcc/doc/extend.texi
> @@ -7617,6 +7617,83 @@ When both the attribute and the option present at the same time, the level of
> the strictness for the specific trailing array field is determined by the
> attribute.
>
> +@cindex @code{counted_by} variable attribute
> +@item counted_by (@var{count})
> +The @code{counted_by} attribute may be attached to the flexible array
> +member of a structure. It indicates that the number of the elements of the
> +array is given by the field named "@var{count}" in the same structure as the
> +flexible array member. GCC uses this information to improve the results of
> +the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
Maybe specify somehow that this only applies to C99 flexible arrays? Like:
The @code{counted_by} attribute may be attached to the C99 flexible
array member of a structure...
> +
> +For instance, the following code:
> +
> +@smallexample
> +struct P @{
> + size_t count;
> + char other;
> + char array[] __attribute__ ((counted_by (count)));
> +@} *p;
> +@end smallexample
> +
> +@noindent
> +specifies that the @code{array} is a flexible array member whose number of
> +elements is given by the field @code{count} in the same structure.
> +
> +The field that represents the number of the elements should have an integer
> +type. An explicit @code{counted_by} annotation defines a relationship between
> +two objects, @code{p->array} and @code{p->count}, that @code{p->array} has
> +@emph{at least} @code{p->count} number of elements available. This relationship
> +must hold even after any of these related objects are updated. It's the user's
> +responsibility to make sure this relationship to be kept all the time.
> +Otherwise the results of the array bound sanitizer and the
> +@code{__builtin_dynamic_object_size} might be incorrect.
Suggest:
The field that represents the number of the elements must have an
integer type. An explicit @code{counted_by} annotation defines a
relationship between two objects, @code{p->array} and @code{p->count},
that @code{p->array} has @emph{at least} @code{p->count} number of
elements available. The user is responsible to ensure that this
relationship is consistently maintained during the lifetime of the
object. Failure to do so may result in results of the array bound
sanitizer and the @code{__builtin_dynamic_object_size} being
undefined.
> +
> +For instance, in the following example, the allocated array has less elements
> +than what's specified by the @code{sbuf->count}, this is an user error. As a
> +result, out-of-bounds access to the array might not be detected.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + nelems * sizeof (char))));
> + sbuf->count = nelems + SIZE_BUMP;
> + /* This is invalid when the sbuf->array has less than sbuf->count
> + elements. */
> +@}
> +@end smallexample
> +
> +In the following example, the 2nd update to the field @code{sbuf->count} of
> +the above structure will permit out-of-bounds access to the array
> +@code{sbuf>array} as well.
> +
> +@smallexample
> +#define SIZE_BUMP 10
> +struct P *sbuf;
> +void alloc_buf (size_t nelems)
> +@{
> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
> + (offsetof (struct P, array[0])
> + + (nelems + SIZE_BUMP) * sizeof (char))));
> + sbuf->count = nelems;
> + /* This is valid when the sbuf->array has at least sbuf->count
> + elements. */
> +@}
> +void use_buf (int index)
> +@{
> + sbuf->count = sbuf->count + SIZE_BUMP + 1;
> + /* Now the value of sbuf->count is larger than the number
> + of elements of sbuf->array. */
> + sbuf->array[index] = 0;
> + /* then the out-of-bound access to this array
> + might not be detected. */
> +@}
> +@end smallexample
> +
> +
I'm unsure if we should be demonstrating example code with undefined
behaviour. Maybe the documentation is better off without it.
> @cindex @code{alloc_size} variable attribute
> @item alloc_size (@var{position})
> @itemx alloc_size (@var{position-1}, @var{position-2})
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> new file mode 100644
> index 000000000000..f8ce9776bf86
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
> @@ -0,0 +1,40 @@
> +/* testing the correct usage of attribute counted_by. */
> +/* { dg-do compile } */
> +/* { dg-options "-O2" } */
> +
> +#include <wchar.h>
> +
> +int size;
> +int x __attribute ((counted_by (size))); /* { dg-error "attribute may not be specified for non-field declaration" } */
> +
> +struct trailing {
> + int count;
> + int field __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non-array field" } */
> +};
> +
> +struct trailing_1 {
> + int count;
> + int array_1[0] __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non flexible array member field" } */
> +};
> +
> +int count;
> +struct trailing_array_2 {
> + int count;
> + int array_2[] __attribute ((counted_by ("count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_3 {
> + int other;
> + int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error "argument not an identifier" } */
> +};
> +
> +struct trailing_array_4 {
> + int other;
> + int array_4[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration in the same structure, ignore it" } */
> +};
> +
> +int count;
> +struct trailing_array_5 {
> + float count;
> + int array_5[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration with integer type, ignore it" } */
> +};
Tests look OK in principle, but may need updating for the error message.
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index 420857b110c4..fcd36ae0cd74 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> + with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> + whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> + return NULL_TREE if such field is not found. Normally this list is of
> + length one, but if the field is embedded with (nested) anonymous structures
> + or unions, this list steps down the chain to the field. */
> +tree
> +get_named_field (tree fieldlist, const char *fieldname)
> +{
> + tree named_field = NULL_TREE;
> + for (tree field = fieldlist; field; field = DECL_CHAIN (field))
> + {
> + if (TREE_CODE (field) != FIELD_DECL)
> + continue;
> + if (DECL_NAME (field) != NULL)
> + if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + /* if the field is an anonymous struct/union, we will check the nested
> + fields inside it recursively. */
> + else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> + if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> + fieldname)) != NULL_TREE)
> + {
> + named_field = tree_cons (NULL_TREE, field, named_field);
> + break;
> + }
> + else
> + continue;
> + else
> + continue;
> + }
> + return named_field;
> +}
Descending recursively into the field list of a struct to find an
identifier with a matching name. OK.
> +
> +
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4c04245e2b1b..4859becaa1e7 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> + name is FIELDNAME, return NULL_TREE if such field is not found.
> + searching nested anonymous structure/union recursively. */
> +extern tree get_named_field (tree, const char *);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-10-05 18:51 ` Siddhesh Poyarekar
@ 2023-10-05 19:31 ` Siddhesh Poyarekar
2023-10-18 14:51 ` Qing Zhao
2023-10-18 14:41 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-05 19:31 UTC (permalink / raw)
To: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, uecker, isanbard
On 2023-10-05 14:51, Siddhesh Poyarekar wrote:
> On 2023-08-25 11:24, Qing Zhao wrote:
>> Provide a new counted_by attribute to flexible array member field.
>
> The obligatory "I can't ack the patch but here's a review" disclaimer :)
>
>>
>> 'counted_by (COUNT)'
>> The 'counted_by' attribute may be attached to the flexible array
>> member of a structure. It indicates that the number of the
>> elements of the array is given by the field named "COUNT" in the
>> same structure as the flexible array member. GCC uses this
>> information to improve the results of the array bound sanitizer and
>> the '__builtin_dynamic_object_size'.
>>
>> For instance, the following code:
>>
>> struct P {
>> size_t count;
>> char other;
>> char array[] __attribute__ ((counted_by (count)));
>> } *p;
>>
>> specifies that the 'array' is a flexible array member whose number
>> of elements is given by the field 'count' in the same structure.
>>
>> The field that represents the number of the elements should have an
>> integer type. An explicit 'counted_by' annotation defines a
>> relationship between two objects, 'p->array' and 'p->count', that
>> 'p->array' has _at least_ 'p->count' number of elements available.
>> This relationship must hold even after any of these related objects
>> are updated. It's the user's responsibility to make sure this
>> relationship to be kept all the time. Otherwise the results of the
>> array bound sanitizer and the '__builtin_dynamic_object_size' might
>> be incorrect.
>>
>> For instance, in the following example, the allocated array has
>> less elements than what's specified by the 'sbuf->count', this is
>> an user error. As a result, out-of-bounds access to the array
>> might not be detected.
>>
>> #define SIZE_BUMP 10
>> struct P *sbuf;
>> void alloc_buf (size_t nelems)
>> {
>> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> (offsetof (struct P,
>> array[0])
>> + nelems * sizeof (char))));
>> sbuf->count = nelems + SIZE_BUMP;
>> /* This is invalid when the sbuf->array has less than
>> sbuf->count
>> elements. */
>> }
>>
>> In the following example, the 2nd update to the field 'sbuf->count'
>> of the above structure will permit out-of-bounds access to the
>> array 'sbuf>array' as well.
>>
>> #define SIZE_BUMP 10
>> struct P *sbuf;
>> void alloc_buf (size_t nelems)
>> {
>> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> (offsetof (struct P,
>> array[0])
>> + (nelems + SIZE_BUMP) *
>> sizeof (char))));
>> sbuf->count = nelems;
>> /* This is valid when the sbuf->array has at least
>> sbuf->count
>> elements. */
>> }
>> void use_buf (int index)
>> {
>> sbuf->count = sbuf->count + SIZE_BUMP + 1;
>> /* Now the value of sbuf->count is larger than the number
>> of elements of sbuf->array. */
>> sbuf->array[index] = 0;
>> /* then the out-of-bound access to this array
>> might not be detected. */
>> }
>>
>> gcc/c-family/ChangeLog:
>>
>> PR C/108896
>> * c-attribs.cc (handle_counted_by_attribute): New function.
>> (attribute_takes_identifier_p): Add counted_by attribute to the list.
>> * c-common.cc (c_flexible_array_member_type_p): ...To this.
>> * c-common.h (c_flexible_array_member_type_p): New prototype.
>>
>> gcc/c/ChangeLog:
>>
>> PR C/108896
>> * c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
>> (add_flexible_array_elts_to_size): Use renamed function.
>> (is_flexible_array_member_p): Use renamed function.
>> (verify_counted_by_attribute): New function.
>> (finish_struct): Use renamed function and verify counted_by
>> attribute.
>>
>> gcc/ChangeLog:
>>
>> PR C/108896
>> * doc/extend.texi: Document attribute counted_by.
>> * tree.cc (get_named_field): New function.
>> * tree.h (get_named_field): New prototype.
>>
>> gcc/testsuite/ChangeLog:
>>
>> PR C/108896
>> * gcc.dg/flex-array-counted-by.c: New test.
>> ---
>> gcc/c-family/c-attribs.cc | 54 ++++++++++++-
>> gcc/c-family/c-common.cc | 13 ++++
>> gcc/c-family/c-common.h | 1 +
>> gcc/c/c-decl.cc | 79 +++++++++++++++-----
>> gcc/doc/extend.texi | 77 +++++++++++++++++++
>> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
>> gcc/tree.cc | 40 ++++++++++
>> gcc/tree.h | 5 ++
>> 8 files changed, 291 insertions(+), 18 deletions(-)
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>>
>> diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
>> index e2792ca6898b..65e4f6639109 100644
>> --- a/gcc/c-family/c-attribs.cc
>> +++ b/gcc/c-family/c-attribs.cc
>> @@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute
>> (tree *, tree, tree,
>> int, bool *);
>> static tree handle_strict_flex_array_attribute (tree *, tree, tree,
>> int, bool *);
>> +static tree handle_counted_by_attribute (tree *, tree, tree,
>> + int, bool *);
>> static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
>> static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
>> static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree,
>> bool *);
>> @@ -373,6 +375,8 @@ const struct attribute_spec
>> c_common_attribute_table[] =
>> handle_warn_if_not_aligned_attribute, NULL },
>> { "strict_flex_array", 1, 1, true, false, false, false,
>> handle_strict_flex_array_attribute, NULL },
>> + { "counted_by", 1, 1, true, false, false, false,
>> + handle_counted_by_attribute, NULL },
>> { "weak", 0, 0, true, false, false, false,
>> handle_weak_attribute, NULL },
>> { "noplt", 0, 0, true, false, false, false,
>> @@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
>> else if (!strcmp ("mode", spec->name)
>> || !strcmp ("format", spec->name)
>> || !strcmp ("cleanup", spec->name)
>> - || !strcmp ("access", spec->name))
>> + || !strcmp ("access", spec->name)
>> + || !strcmp ("counted_by", spec->name))
>> return true;
>> else
>> return targetm.attribute_takes_identifier_p (attr_id);
>> @@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node,
>> tree name,
>> return NULL_TREE;
>> }
>> +/* Handle a "counted_by" attribute; arguments as in
>> + struct attribute_spec.handler. */
>> +
>> +static tree
>> +handle_counted_by_attribute (tree *node, tree name,
>> + tree args, int ARG_UNUSED (flags),
>> + bool *no_add_attrs)
>> +{
>> + tree decl = *node;
>> + tree argval = TREE_VALUE (args);
>> +
>> + /* This attribute only applies to field decls of a structure. */
>> + if (TREE_CODE (decl) != FIELD_DECL)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for non-field"
>> + " declaration %q+D", name, decl);
>> + *no_add_attrs = true;
>> + }
>
> Applies only to struct fields. OK.
>
>> + /* This attribute only applies to field with array type. */
>> + else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for a non-array field",
>> + name);
>> + *no_add_attrs = true;
>> + }
>
> The struct field should also be an array. OK.
>
>> + /* This attribute only applies to a C99 flexible array member
>> type. */
>> + else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for a non"
>> + " flexible array member field",
>> + name);
>> + *no_add_attrs = true;
>> + }
>
> Additionally, the field should be a *flex* array. OK. Could this be
> reworded to:
>
> %qE attribute only applies to C99 flexible array members
>
> That would make it clear that the GNU extension flex arrays (i.e. any
> last array member of a struct) don't support this attribute.
>
>> + /* The argument should be an identifier. */
>> + else if (TREE_CODE (argval) != IDENTIFIER_NODE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%<counted_by%> argument not an identifier");
>> + *no_add_attrs = true;
>> + }
>
> Argument should be an identifier, and the check for validity of the
> identifier comes later in finish_struct. OK.
>
>> +
>> + return NULL_TREE;
>> +}
>> +
>> /* Handle a "weak" attribute; arguments as in
>> struct attribute_spec.handler. */
>> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
>> index 9fbaeb437a12..a18937245c2a 100644
>> --- a/gcc/c-family/c-common.cc
>> +++ b/gcc/c-family/c-common.cc
>> @@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
>> (*debug_hooks->early_global_decl) (cnode->decl);
>> }
>> +/* Determine whether TYPE is a ISO C99 flexible array memeber type
>> "[]". */
>> +bool
>> +c_flexible_array_member_type_p (const_tree type)
>> +{
>> + if (TREE_CODE (type) == ARRAY_TYPE
>> + && TYPE_SIZE (type) == NULL_TREE
>> + && TYPE_DOMAIN (type) != NULL_TREE
>> + && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
>> + return true;
>> +
>> + return false;
>> +}
>> +
>
> Hoist flexible_array_member_type_p to use more widely. OK.
>
>> /* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based
>> on the
>> values of attribute strict_flex_array and the
>> flag_strict_flex_arrays. */
>> unsigned int
>> diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
>> index 78fc5248ba68..c29bb429062b 100644
>> --- a/gcc/c-family/c-common.h
>> +++ b/gcc/c-family/c-common.h
>> @@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
>> extern tree c_common_get_narrower (tree, int *);
>> extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
>> extern void c_common_finalize_early_debug (void);
>> +extern bool c_flexible_array_member_type_p (const_tree);
>> extern unsigned int c_strict_flex_array_level_of (tree);
>> extern bool c_option_is_from_cpp_diagnostics (int);
>> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
>> index 1f9eb44dbaa2..e943b49b5230 100644
>> --- a/gcc/c/c-decl.cc
>> +++ b/gcc/c/c-decl.cc
>> @@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator
>> *decl,
>> return decl;
>> }
>> -/* Determine whether TYPE is a ISO C99 flexible array memeber type
>> "[]". */
>> -static bool
>> -flexible_array_member_type_p (const_tree type)
>> -{
>> - if (TREE_CODE (type) == ARRAY_TYPE
>> - && TYPE_SIZE (type) == NULL_TREE
>> - && TYPE_DOMAIN (type) != NULL_TREE
>> - && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
>> - return true;
>> -
>> - return false;
>> -}
>> -
>> /* Determine whether TYPE is a one-element array type "[1]". */
>> static bool
>> one_element_array_type_p (const_tree type)
>> @@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree
>> init)
>> elt = CONSTRUCTOR_ELTS (init)->last ().value;
>> type = TREE_TYPE (elt);
>> - if (flexible_array_member_type_p (type))
>> + if (c_flexible_array_member_type_p (type))
>> {
>> complete_array_type (&type, elt, false);
>> DECL_SIZE (decl)
>> @@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
>> bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
>> bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
>> - bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
>> + bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE
>> (x));
>> unsigned int strict_flex_array_level =
>> c_strict_flex_array_level_of (x);
>
> Simple refactoring. OK.
>
>> @@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
>> return false;
>> }
>> +/* Verify the argument of the counted_by attribute of the flexible array
>
> Verify *that* the argument...
>
>> + member FIELD_DECL is a valid field of the containing structure's
>> fieldlist,
>> + FIELDLIST, Report error and remove this attribute when it's not. */
>> +static void
>> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
>> +{
>> + tree attr_counted_by = lookup_attribute ("counted_by",
>> + DECL_ATTRIBUTES (field_decl));
>> +
>> + if (!attr_counted_by)
>> + return;
>> +
>> + /* If there is an counted_by attribute attached to the field,
>> + verify it. */
>> +
>> + const char *fieldname
>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>> +
>> + /* Verify the argument of the attrbute is a valid field of the
>
> s/attrbute/attribute/
>
>> + containing structure. */
>> +
>> + tree counted_by_field = get_named_field (fieldlist, fieldname);
>> +
>> + /* Error when the field is not found in the containing structure. */
>> + if (!counted_by_field)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>> + "%qE attribute argument not a field declaration"
>> + " in the same structure, ignore it",
>> + (get_attribute_name (attr_counted_by)));
>
> Probably someone with English as a first language would make a better
> suggestion, but how about:
>
> Argument specified in %qE attribute is not a field declaration in the
> same structure, ignoring it.
>
>> +
>> + DECL_ATTRIBUTES (field_decl)
>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>> + }
>> + else
>> + /* Error when the field is not with an integer type. */
>
> Suggest: Flag an error when the field is not of an integer type.
>
>> + {
>> + while (TREE_CHAIN (counted_by_field))
>> + counted_by_field = TREE_CHAIN (counted_by_field);
>> + tree real_field = TREE_VALUE (counted_by_field);
>> +
>> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>> + "%qE attribute argument not a field declaration"
>> + " with integer type, ignore it",
>> + (get_attribute_name (attr_counted_by)));
>
> Suggest:
>
> Argument specified in %qE attribute is not of an integer type,
> ignoring it.
>
>> +
>> + DECL_ATTRIBUTES (field_decl)
>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>> + }
>> + }
>> +
>> + return;
I forgot to mention the redundant return here.
>> +}
>> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
>> LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
>> @@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree
>> fieldlist, tree attributes,
>> DECL_PACKED (x) = 1;
>> /* Detect flexible array member in an invalid context. */
>> - if (flexible_array_member_type_p (TREE_TYPE (x)))
>> + if (c_flexible_array_member_type_p (TREE_TYPE (x)))
>> {
>> if (TREE_CODE (t) == UNION_TYPE)
>> {
>> @@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree
>> fieldlist, tree attributes,
>> "members");
>> TREE_TYPE (x) = error_mark_node;
>> }
>> + /* if there is a counted_by attribute attached to this field,
>> + verify it. */
>> + verify_counted_by_attribute (fieldlist, x);
>> }
>> if (pedantic && TREE_CODE (t) == RECORD_TYPE
>> @@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree
>> fieldlist, tree attributes,
>> when x is an array and is the last field. */
>> if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
>> TYPE_INCLUDES_FLEXARRAY (t)
>> - = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
>> + = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
>> /* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of
>> x, t
>> when x is an union or record and is the last field. */
>> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
>> diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
>> index 97eaacf8a7ec..ea6240646936 100644
>> --- a/gcc/doc/extend.texi
>> +++ b/gcc/doc/extend.texi
>> @@ -7617,6 +7617,83 @@ When both the attribute and the option present
>> at the same time, the level of
>> the strictness for the specific trailing array field is determined
>> by the
>> attribute.
>> +@cindex @code{counted_by} variable attribute
>> +@item counted_by (@var{count})
>> +The @code{counted_by} attribute may be attached to the flexible array
>> +member of a structure. It indicates that the number of the elements
>> of the
>> +array is given by the field named "@var{count}" in the same structure
>> as the
>> +flexible array member. GCC uses this information to improve the
>> results of
>> +the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
>
> Maybe specify somehow that this only applies to C99 flexible arrays? Like:
>
> The @code{counted_by} attribute may be attached to the C99 flexible
> array member of a structure...
>
>> +
>> +For instance, the following code:
>> +
>> +@smallexample
>> +struct P @{
>> + size_t count;
>> + char other;
>> + char array[] __attribute__ ((counted_by (count)));
>> +@} *p;
>> +@end smallexample
>> +
>> +@noindent
>> +specifies that the @code{array} is a flexible array member whose
>> number of
>> +elements is given by the field @code{count} in the same structure.
>> +
>> +The field that represents the number of the elements should have an
>> integer
>> +type. An explicit @code{counted_by} annotation defines a
>> relationship between
>> +two objects, @code{p->array} and @code{p->count}, that
>> @code{p->array} has
>> +@emph{at least} @code{p->count} number of elements available. This
>> relationship
>> +must hold even after any of these related objects are updated. It's
>> the user's
>> +responsibility to make sure this relationship to be kept all the time.
>> +Otherwise the results of the array bound sanitizer and the
>> +@code{__builtin_dynamic_object_size} might be incorrect.
>
> Suggest:
>
> The field that represents the number of the elements must have an
> integer type. An explicit @code{counted_by} annotation defines a
> relationship between two objects, @code{p->array} and @code{p->count},
> that @code{p->array} has @emph{at least} @code{p->count} number of
> elements available. The user is responsible to ensure that this
> relationship is consistently maintained during the lifetime of the
> object. Failure to do so may result in results of the array bound
> sanitizer and the @code{__builtin_dynamic_object_size} being
> undefined.
>
>> +
>> +For instance, in the following example, the allocated array has less
>> elements
>> +than what's specified by the @code{sbuf->count}, this is an user
>> error. As a
>> +result, out-of-bounds access to the array might not be detected.
>> +
>> +@smallexample
>> +#define SIZE_BUMP 10
>> +struct P *sbuf;
>> +void alloc_buf (size_t nelems)
>> +@{
>> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> + (offsetof (struct P, array[0])
>> + + nelems * sizeof (char))));
>> + sbuf->count = nelems + SIZE_BUMP;
>> + /* This is invalid when the sbuf->array has less than sbuf->count
>> + elements. */
>> +@}
>> +@end smallexample
>> +
>> +In the following example, the 2nd update to the field
>> @code{sbuf->count} of
>> +the above structure will permit out-of-bounds access to the array
>> +@code{sbuf>array} as well.
>> +
>> +@smallexample
>> +#define SIZE_BUMP 10
>> +struct P *sbuf;
>> +void alloc_buf (size_t nelems)
>> +@{
>> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> + (offsetof (struct P, array[0])
>> + + (nelems + SIZE_BUMP) * sizeof (char))));
>> + sbuf->count = nelems;
>> + /* This is valid when the sbuf->array has at least sbuf->count
>> + elements. */
>> +@}
>> +void use_buf (int index)
>> +@{
>> + sbuf->count = sbuf->count + SIZE_BUMP + 1;
>> + /* Now the value of sbuf->count is larger than the number
>> + of elements of sbuf->array. */
>> + sbuf->array[index] = 0;
>> + /* then the out-of-bound access to this array
>> + might not be detected. */
>> +@}
>> +@end smallexample
>> +
>> +
>
> I'm unsure if we should be demonstrating example code with undefined
> behaviour. Maybe the documentation is better off without it.
>
>> @cindex @code{alloc_size} variable attribute
>> @item alloc_size (@var{position})
>> @itemx alloc_size (@var{position-1}, @var{position-2})
>> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> new file mode 100644
>> index 000000000000..f8ce9776bf86
>> --- /dev/null
>> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> @@ -0,0 +1,40 @@
>> +/* testing the correct usage of attribute counted_by. */
>> +/* { dg-do compile } */
>> +/* { dg-options "-O2" } */
>> +
>> +#include <wchar.h>
>> +
>> +int size;
>> +int x __attribute ((counted_by (size))); /* { dg-error "attribute may
>> not be specified for non-field declaration" } */
>> +
>> +struct trailing {
>> + int count;
>> + int field __attribute ((counted_by (count))); /* { dg-error
>> "attribute may not be specified for a non-array field" } */
>> +};
>> +
>> +struct trailing_1 {
>> + int count;
>> + int array_1[0] __attribute ((counted_by (count))); /* { dg-error
>> "attribute may not be specified for a non flexible array member field"
>> } */
>> +};
>> +
>> +int count;
>> +struct trailing_array_2 {
>> + int count;
>> + int array_2[] __attribute ((counted_by ("count"))); /* { dg-error
>> "argument not an identifier" } */
>> +};
>> +
>> +struct trailing_array_3 {
>> + int other;
>> + int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error
>> "argument not an identifier" } */
>> +};
>> +
>> +struct trailing_array_4 {
>> + int other;
>> + int array_4[] __attribute ((counted_by (count))); /* { dg-error
>> "attribute argument not a field declaration in the same structure,
>> ignore it" } */
>> +};
>> +
>> +int count;
>> +struct trailing_array_5 {
>> + float count;
>> + int array_5[] __attribute ((counted_by (count))); /* { dg-error
>> "attribute argument not a field declaration with integer type, ignore
>> it" } */
>> +};
>
> Tests look OK in principle, but may need updating for the error message.
>
>> diff --git a/gcc/tree.cc b/gcc/tree.cc
>> index 420857b110c4..fcd36ae0cd74 100644
>> --- a/gcc/tree.cc
>> +++ b/gcc/tree.cc
>> @@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
>> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT
>> (elmt_type), exp);
>> }
>> +/* Given a field list, FIELDLIST, of a structure/union, return a
>> TREE_LIST,
>> + with each TREE_VALUE a FIELD_DECL stepping down the chain to the
>> FIELD
>> + whose name is FIELDNAME, which is the last TREE_VALUE of the list.
>> + return NULL_TREE if such field is not found. Normally this list
>> is of
>> + length one, but if the field is embedded with (nested) anonymous
>> structures
>> + or unions, this list steps down the chain to the field. */
>> +tree
>> +get_named_field (tree fieldlist, const char *fieldname)
>> +{
>> + tree named_field = NULL_TREE;
>> + for (tree field = fieldlist; field; field = DECL_CHAIN (field))
>> + {
>> + if (TREE_CODE (field) != FIELD_DECL)
>> + continue;
>> + if (DECL_NAME (field) != NULL)
>> + if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
>> + {
>> + named_field = tree_cons (NULL_TREE, field, named_field);
>> + break;
>> + }
>> + else
>> + continue;
>> + /* if the field is an anonymous struct/union, we will check the
>> nested
>> + fields inside it recursively. */
>> + else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
>> + if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE
>> (field)),
>> + fieldname)) != NULL_TREE)
>> + {
>> + named_field = tree_cons (NULL_TREE, field, named_field);
>> + break;
>> + }
>> + else
>> + continue;
>> + else
>> + continue;
>> + }
>> + return named_field;
>> +}
>
> Descending recursively into the field list of a struct to find an
> identifier with a matching name. OK.
>
>> +
>> +
>> /* Return a tree representing the lower bound of the array mentioned in
>> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> diff --git a/gcc/tree.h b/gcc/tree.h
>> index 4c04245e2b1b..4859becaa1e7 100644
>> --- a/gcc/tree.h
>> +++ b/gcc/tree.h
>> @@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
>> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> extern tree array_ref_element_size (tree);
>> +/* Given a field list, FIELDLIST, of a structure/union, return the
>> FIELD whose
>> + name is FIELDNAME, return NULL_TREE if such field is not found.
>> + searching nested anonymous structure/union recursively. */
>> +extern tree get_named_field (tree, const char *);
>> +
>> /* Return a typenode for the "standard" C type with a given name. */
>> extern tree get_typenode_from_name (const char *);
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896]
2023-08-25 15:24 ` [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896] Qing Zhao
2023-09-08 14:12 ` Qing Zhao
2023-09-20 13:44 ` PING *2: " Qing Zhao
@ 2023-10-05 20:01 ` Siddhesh Poyarekar
2023-10-18 20:39 ` Qing Zhao
2 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-05 20:01 UTC (permalink / raw)
To: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, uecker, isanbard
On 2023-08-25 11:24, Qing Zhao wrote:
> Use the counted_by atribute info in builtin object size to compute the
> subobject size for flexible array members.
>
> gcc/ChangeLog:
>
> PR C/108896
> * tree-object-size.cc (addr_object_size): Use the counted_by
> attribute info.
> * tree.cc (component_ref_has_counted_by_p): New function.
> (component_ref_get_counted_by): New function.
> * tree.h (component_ref_has_counted_by_p): New prototype.
> (component_ref_get_counted_by): New prototype.
>
> gcc/testsuite/ChangeLog:
>
> PR C/108896
> * gcc.dg/flex-array-counted-by-2.c: New test.
> * gcc.dg/flex-array-counted-by-3.c: New test.
> ---
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 95 +++++++-
> gcc/tree.h | 10 +
> 5 files changed, 418 insertions(+), 8 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> new file mode 100644
> index 000000000000..ec580c1f1f01
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> @@ -0,0 +1,74 @@
> +/* test the attribute counted_by and its usage in
> + * __builtin_dynamic_object_size. */
> +/* { dg-do run } */
> +/* { dg-options "-O2" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
You're using this in a bunch of tests already; does it make sense to
consolidate it into builtin-object-size-common.h?
> +
> +struct flex {
> + int b;
> + int c[];
> +} *array_flex;
> +
> +struct annotated {
> + int b;
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_annotated;
> +
> +struct nested_annotated {
> + struct {
> + union {
> + int b;
> + float f;
> + };
> + int n;
> + };
> + int c[] __attribute__ ((counted_by (b)));
> +} *array_nested_annotated;
> +
> +void __attribute__((__noinline__)) setup (int normal_count, int attr_count)
> +{
> + array_flex
> + = (struct flex *)malloc (sizeof (struct flex)
> + + normal_count * sizeof (int));
> + array_flex->b = normal_count;
> +
> + array_annotated
> + = (struct annotated *)malloc (sizeof (struct annotated)
> + + attr_count * sizeof (int));
> + array_annotated->b = attr_count;
> +
> + array_nested_annotated
> + = (struct nested_annotated *)malloc (sizeof (struct nested_annotated)
> + + attr_count * sizeof (int));
> + array_nested_annotated->b = attr_count;
> +
> + return;
> +}
> +
> +void __attribute__((__noinline__)) test ()
> +{
> + expect(__builtin_dynamic_object_size(array_flex->c, 1), -1);
> + expect(__builtin_dynamic_object_size(array_annotated->c, 1),
> + array_annotated->b * sizeof (int));
> + expect(__builtin_dynamic_object_size(array_nested_annotated->c, 1),
> + array_nested_annotated->b * sizeof (int));
> +}
Maybe another test where the allocation, size assignment and __bdos call
happen in the same function, where the allocator is not recognized by gcc:
void *
__attribute__ ((noinline))
alloc (size_t sz)
{
return __builtin_malloc (sz);
}
void test (size_t sz)
{
array_annotated = alloc (sz);
array_annotated->b = sz;
return __builtin_dynamic_object_size (array_annotated->c, 1);
}
The interesting thing to test (and ensure in the codegen) is that the
assignment to array_annotated->b does not get reordered to below the
__builtin_dynamic_object_size call since technically there is no data
dependency between the two.
> +
> +int main(int argc, char *argv[])
> +{
> + setup (10,10);
> + test ();
> + DONE ();
> +}
> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> new file mode 100644
> index 000000000000..a0c3cb88ec71
> --- /dev/null
> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> @@ -0,0 +1,210 @@
> +/* test the attribute counted_by and its usage in
> +__builtin_dynamic_object_size: what's the correct behavior when the
> +allocation size mismatched with the value of counted_by attribute? */
If the behaviour is undefined, does it make sense to add tests for this?
Maybe once you have a -Wmismatched-counted-by or similar, we could
have tests for that. I guess the counter-argument is that we keep track
of this behaviour but not necessarily guarantee it.
> +/* { dg-do run } */
> +/* { dg-options "-O -fstrict-flex-arrays=3" } */
> +
> +#include "builtin-object-size-common.h"
> +
> +struct annotated {
> + size_t foo;
> + char others;
> + char array[] __attribute__((counted_by (foo)));
> +};
> +
> +#define expect(p, _v) do { \
> + size_t v = _v; \
> + if (p == v) \
> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
> + else \
> + { \
> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
> + FAIL (); \
> + } \
> +} while (0);
Same, maybe consolidate this into builtin-object-size-common.h.
> +
> +#define noinline __attribute__((__noinline__))
> +#define SIZE_BUMP 10
> +#define MAX(a, b) ((a) > (b) ? (a) : (b))
> +#define MIN(a, b) ((a) < (b) ? (a) : (b))
> +
> +/* In general, Due to type casting, the type for the pointee of a pointer
> + does not say anything about the object it points to,
> + So, __builtin_object_size can not directly use the type of the pointee
> + to decide the size of the object the pointer points to.
> +
> + there are only two reliable ways:
> + A. observed allocations (call to the allocation functions in the routine)
> + B. observed accesses (read or write access to the location of the
> + pointer points to)
> +
> + that provide information about the type/existence of an object at
> + the corresponding address.
> +
> + for A, we use the "alloc_size" attribute for the corresponding allocation
> + functions to determine the object size;
> +
> + For B, we use the SIZE info of the TYPE attached to the corresponding access.
> + (We treat counted_by attribute as a complement to the SIZE info of the TYPE
> + for FMA)
> +
> + The only other way in C which ensures that a pointer actually points
> + to an object of the correct type is 'static':
> +
> + void foo(struct P *p[static 1]);
> +
> + See https://gcc.gnu.org/pipermail/gcc-patches/2023-July/624814.html
> + for more details. */
> +
> +/* in the following function, malloc allocated more space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases? */
> +
> +static struct annotated * noinline alloc_buf_more (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index + SIZE_BUMP) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed allocation.
> + So, the object size info can only been obtained from the call to malloc.
> + for both MAXIMUM and MINIMUM: A = (index + SIZE_BUMP) * sizeof (char) */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +/* in the following function, malloc allocated less space than the value
> + of counted_by attribute. Then what's the correct behavior we expect
> + the __builtin_dynamic_object_size should have for each of the cases?
> + NOTE: this is an user error, GCC should issue warnings for such case.
> + this is a seperate issue we should address later. */
> +
> +static struct annotated * noinline alloc_buf_less (size_t index)
> +{
> + struct annotated *p;
> + size_t allocated_size
> + = MAX (sizeof (struct annotated),
> + (__builtin_offsetof (struct annotated, array[0])
> + + (index) * sizeof (char)));
> + p = (struct annotated *) malloc (allocated_size);
> +
> + p->foo = index + SIZE_BUMP;
> +
> + /*when checking the observed access p->array, we have info on both
> + observered allocation and observed access,
> + A. from observed allocation:
> + allocated_size - offsetof (struct annotated, array[0])
> + B. from observed access: p->foo * sizeof (char)
> + */
> +
> + /* for size in the whole object: always uses A. */
> + /* for size in the sub-object: chose the smaller of A and B.
> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
> + * for details on why. */
> +
> + /* for MAXIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 1),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /* for MINIMUM size in the whole object: use the allocation size
> + for the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 2),
> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
> +
> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
> + expect(__builtin_dynamic_object_size(p->array, 3),
> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
> + (p->foo) * sizeof(char)));
> +
> + /*when checking the pointer p, we only have info on the observed
> + allocation. So, the object size info can only been obtained from
> + the call to malloc. */
> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
> + return p;
> +}
> +
> +int main ()
> +{
> + struct annotated *p, *q;
> + p = alloc_buf_more (10);
> + q = alloc_buf_less (10);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(p->array, 0), -1);
> + expect(__builtin_dynamic_object_size(p->array, 1), p->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(p->array, 2), 0);
> + expect(__builtin_dynamic_object_size(p->array, 3), p->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(p, 0), -1);
> + expect(__builtin_dynamic_object_size(p, 1), -1);
> + expect(__builtin_dynamic_object_size(p, 2), 0);
> + expect(__builtin_dynamic_object_size(p, 3), 0);
> +
> + /*when checking the observed access p->array, we only have info on the
> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
> + info on the whole object. */
> + expect(__builtin_dynamic_object_size(q->array, 0), -1);
> + expect(__builtin_dynamic_object_size(q->array, 1), q->foo * sizeof(char));
> + expect(__builtin_dynamic_object_size(q->array, 2), 0);
> + expect(__builtin_dynamic_object_size(q->array, 3), q->foo * sizeof(char));
> + /*when checking the pointer p, we have no observed allocation nor observed
> + access, therefore, we cannot determine the size info here. */
> + expect(__builtin_dynamic_object_size(q, 0), -1);
> + expect(__builtin_dynamic_object_size(q, 1), -1);
> + expect(__builtin_dynamic_object_size(q, 2), 0);
> + expect(__builtin_dynamic_object_size(q, 3), 0);
> +
> + DONE ();
> +}
> diff --git a/gcc/tree-object-size.cc b/gcc/tree-object-size.cc
> index a62af0500563..cf7843c5684b 100644
> --- a/gcc/tree-object-size.cc
> +++ b/gcc/tree-object-size.cc
> @@ -585,6 +585,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> if (pt_var != TREE_OPERAND (ptr, 0))
> {
> tree var;
> + tree counted_by_ref = NULL_TREE;
>
> if (object_size_type & OST_SUBOBJECT)
> {
> @@ -600,11 +601,12 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> var = TREE_OPERAND (var, 0);
> if (var != pt_var && TREE_CODE (var) == ARRAY_REF)
> var = TREE_OPERAND (var, 0);
> - if (! TYPE_SIZE_UNIT (TREE_TYPE (var))
> + if (! component_ref_has_counted_by_p (var)
> + && ((! TYPE_SIZE_UNIT (TREE_TYPE (var))
> || ! tree_fits_uhwi_p (TYPE_SIZE_UNIT (TREE_TYPE (var)))
> || (pt_var_size && TREE_CODE (pt_var_size) == INTEGER_CST
> && tree_int_cst_lt (pt_var_size,
> - TYPE_SIZE_UNIT (TREE_TYPE (var)))))
> + TYPE_SIZE_UNIT (TREE_TYPE (var)))))))
> var = pt_var;
> else if (var != pt_var && TREE_CODE (pt_var) == MEM_REF)
> {
Hmm, only for subobject size? I thought we had consensus on using
sizeof (struct) + counted_by_size as the conservative maximum size for
whole object size too, didn't we?
> @@ -612,6 +614,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* For &X->fld, compute object size if fld isn't a flexible array
> member. */
> bool is_flexible_array_mem_ref = false;
> +
> while (v && v != pt_var)
> switch (TREE_CODE (v))
> {
Unnecessary newline.
> @@ -660,6 +663,8 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> /* Now the ref is to an array type. */
> gcc_assert (TREE_CODE (TREE_TYPE (v)) == ARRAY_TYPE);
> is_flexible_array_mem_ref = array_ref_flexible_size_p (v);
> + counted_by_ref = component_ref_get_counted_by (v);
> +
> while (v != pt_var && TREE_CODE (v) == COMPONENT_REF)
> if (TREE_CODE (TREE_TYPE (TREE_OPERAND (v, 0)))
> != UNION_TYPE
> @@ -673,8 +678,11 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
> == RECORD_TYPE)
> {
> /* compute object size only if v is not a
> - flexible array member. */
> - if (!is_flexible_array_mem_ref)
> + flexible array member or the flexible array member
> + has a known element count indicated by the user
> + through attribute counted_by. */
> + if (!is_flexible_array_mem_ref
> + || counted_by_ref)
> {
> v = NULL_TREE;
> break;
> @@ -707,9 +715,24 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>
> if (var != pt_var)
> {
> - var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> - if (!TREE_CONSTANT (var_size))
> - var_size = get_or_create_ssa_default_def (cfun, var_size);
> + if (!counted_by_ref)
> + {
> + var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
> + if (!TREE_CONSTANT (var_size))
> + var_size = get_or_create_ssa_default_def (cfun, var_size);
> + }
> + else
> + {
> + gcc_assert (TREE_CODE (var) == COMPONENT_REF
> + && TREE_CODE (TREE_TYPE (var)) == ARRAY_TYPE);
> + tree element_size = TYPE_SIZE_UNIT (TREE_TYPE (TREE_TYPE (var)));
> + var_size
> + = size_binop (MULT_EXPR,
> + fold_convert (sizetype, counted_by_ref),
> + fold_convert (sizetype, element_size));
> + if (!todo)
> + todo = TODO_update_ssa_only_virtuals;
> + }
I feel like this could make a good separate function (get_subobject_size
or something like that) to make it easier to read.
> if (!var_size)
> return false;
> }
> diff --git a/gcc/tree.cc b/gcc/tree.cc
> index fcd36ae0cd74..3b6ddcbdcbf8 100644
> --- a/gcc/tree.cc
> +++ b/gcc/tree.cc
> @@ -12745,6 +12745,32 @@ array_ref_element_size (tree exp)
> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, return TRUE when
> + an counted_by attribute attached to the corresponding FIELD_DECL.
> + return FALSE otherwise. */
> +bool
> +component_ref_has_counted_by_p (tree array_ref)
> +{
> + if (TREE_CODE (array_ref) != COMPONENT_REF)
> + return false;
> +
> + if (TREE_CODE (TREE_TYPE (array_ref)) != ARRAY_TYPE)
> + return false;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> +
> + if (!RECORD_OR_UNION_TYPE_P (struct_type))
> + return false;
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> +
> + if (!attr_counted_by)
> + return false;
> + return true;
> +}
> +
> /* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
> with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
> whose name is FIELDNAME, which is the last TREE_VALUE of the list.
> @@ -12771,7 +12797,7 @@ get_named_field (tree fieldlist, const char *fieldname)
> fields inside it recursively. */
> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
> if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
> - fieldname)) != NULL_TREE)
> + fieldname)) != NULL_TREE)
Unrelated whitespace change?
> {
> named_field = tree_cons (NULL_TREE, field, named_field);
> break;
> @@ -12784,6 +12810,73 @@ get_named_field (tree fieldlist, const char *fieldname)
> return named_field;
> }
>
> +/* For a component_ref that has an array type ARRAY_REF, get the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object.
> + For example, if:
> +
> + struct P {
> + int k;
> + int x[] __attribute__ ((counted_by (k)));
> + } *p;
> +
> + for the following reference:
> +
> + p->x[b]
> +
> + the object that represents its element count will be:
> +
> + p->k
> +
> + So, when component_ref_get_counted_by (p->x[b]) is called, p->k should be
> + returned.
> +*/
> +
> +tree
> +component_ref_get_counted_by (tree array_ref)
> +{
> + if (! component_ref_has_counted_by_p (array_ref))
> + return NULL_TREE;
> +
> + tree struct_object = TREE_OPERAND (array_ref, 0);
> + tree struct_type = TREE_TYPE (struct_object);
> + tree field_decl = TREE_OPERAND (array_ref, 1);
> + tree attr_counted_by = lookup_attribute ("counted_by",
> + DECL_ATTRIBUTES (field_decl));
> + gcc_assert (attr_counted_by);
> +
> + /* If there is an counted_by attribute attached to the field,
> + get the field that maps to the counted_by. */
> +
> + const char *fieldname
> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
> +
> + tree counted_by_field = get_named_field (TYPE_FIELDS (struct_type),
> + fieldname);
> +
> + gcc_assert (counted_by_field);
> +
> + /* generate the tree node that represent the counted_by of this array
Capitalize first word. Also s/represent/represents/
> + ref. This is a (possible nested) COMPONENT_REF to the counted_by_field
possibly nested
> + of the containing structure. */
> +
> + tree counted_by_ref = NULL_TREE;
> + tree object = struct_object;
> + do
> + {
> + tree field = TREE_VALUE (counted_by_field);
> +
> + counted_by_ref = build3 (COMPONENT_REF,
> + TREE_TYPE (field),
> + unshare_expr (object), field,
> + NULL_TREE);
> + object = counted_by_ref;
> + counted_by_field = TREE_CHAIN (counted_by_field);
> + }
> + while (counted_by_field);
> + return counted_by_ref;
> +}
>
> /* Return a tree representing the lower bound of the array mentioned in
> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> diff --git a/gcc/tree.h b/gcc/tree.h
> index 4859becaa1e7..07eed7219835 100644
> --- a/gcc/tree.h
> +++ b/gcc/tree.h
> @@ -5619,11 +5619,21 @@ extern tree get_base_address (tree t);
> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
> extern tree array_ref_element_size (tree);
>
> +/* Give a component_ref that has an array type, return true when an
> + attribute counted_by attached to the corresponding FIELD_DECL. */
> +extern bool component_ref_has_counted_by_p (tree);
> +
> /* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
> name is FIELDNAME, return NULL_TREE if such field is not found.
> searching nested anonymous structure/union recursively. */
> extern tree get_named_field (tree, const char *);
>
> +/* Give a component_ref that has an array type, return the object that
> + represents its counted_by per the attribute counted_by attached to
> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
> + object. */
> +extern tree component_ref_get_counted_by (tree);
> +
> /* Return a typenode for the "standard" C type with a given name. */
> extern tree get_typenode_from_name (const char *);
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-08-25 15:24 [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896) Qing Zhao
` (5 preceding siblings ...)
2023-09-20 13:43 ` PING * 2: " Qing Zhao
@ 2023-10-05 20:08 ` Siddhesh Poyarekar
2023-10-05 22:35 ` Kees Cook
2023-10-18 21:11 ` Qing Zhao
6 siblings, 2 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-05 20:08 UTC (permalink / raw)
To: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches
Cc: keescook, uecker, isanbard
On 2023-08-25 11:24, Qing Zhao wrote:
> This is the 3rd version of the patch, per our discussion based on the
> review comments for the 1st and 2nd version, the major changes in this
> version are:
Hi Qing,
I hope the review was helpful. Overall, a couple of things to consider:
1. How would you handle potential reordering between assignment of the
size to the counted_by field with the __bdos call that may consume it?
You'll probably need to express some kind of dependency there or in the
worst case, insert a barrier to disallow reordering.
2. How would you handle signedness of the size field? The size gets
converted to sizetype everywhere it is used and overflows/underflows may
produce interesting results. Do you want to limit the types to unsigned
or do you want to add a disclaimer in the docs? The former seems like
the *right* thing to do given that it is a new feature; best to enforce
the cleaner habit at the outset.
Thanks,
Sid
>
> ***Against 1st version:
> 1. change the name "element_count" to "counted_by";
> 2. change the parameter for the attribute from a STRING to an
> Identifier;
> 3. Add logic and testing cases to handle anonymous structure/unions;
> 4. Clarify documentation to permit the situation when the allocation
> size is larger than what's specified by "counted_by", at the same time,
> it's user's error if allocation size is smaller than what's specified by
> "counted_by";
> 5. Add a complete testing case for using counted_by attribute in
> __builtin_dynamic_object_size when there is mismatch between the
> allocation size and the value of "counted_by", the expecting behavior
> for each case and the explanation on why in the comments.
>
> ***Against 2rd version:
> 1. Identify a tree node sharing issue and fixed it in the routine
> "component_ref_get_counted_ty" of tree.cc;
> 2. Update the documentation and testing cases with the clear usage
> of the fomula to compute the allocation size:
> MAX (sizeof (struct A), offsetof (struct A, array[0]) + counted_by * sizeof(element))
> (the algorithm used in tree-object-size.cc is correct).
>
> In this set of patches, the major functionality provided is:
>
> 1. a new attribute "counted_by";
> 2. use this new attribute in bound sanitizer;
> 3. use this new attribute in dynamic object size for subobject size;
>
> As discussed, I plan to add two more separate patches sets after this initial
> patch set is approved and committed.
>
> set 1. A new warning option and a new sanitizer option for the user error
> when the allocation size is smaller than the value of "counted_by".
> set 2. An improvement to __builtin_dynamic_object_size for whole-object
> size of the structure with FAM annaoted with counted_by.
>
> there are also some existing bugs in tree-object-size.cc identified
> during the study, and PRs were filed to record them. these bugs will
> be fixed seperately with individual patches:
>
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111030
> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111040
>
> Bootstrapped and regression tested on both aarch64 and X86, no issue.
>
> Please see more details on the description of this work on:
>
> https://gcc.gnu.org/pipermail/gcc-patches/2023-May/619708.html
>
> and more discussions on
> https://gcc.gnu.org/pipermail/gcc-patches/2023-August/626376.html
>
> Okay for committing?
>
> thanks.
>
> Qing
>
> Qing Zhao (3):
> Provide counted_by attribute to flexible array member field (PR108896)
> Use the counted_by atribute info in builtin object size [PR108896]
> Use the counted_by attribute information in bound sanitizer[PR108896]
>
> gcc/c-family/c-attribs.cc | 54 ++++-
> gcc/c-family/c-common.cc | 13 ++
> gcc/c-family/c-common.h | 1 +
> gcc/c-family/c-ubsan.cc | 16 ++
> gcc/c/c-decl.cc | 79 +++++--
> gcc/doc/extend.texi | 77 +++++++
> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++
> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++
> .../ubsan/flex-array-counted-by-bounds.c | 46 ++++
> gcc/tree-object-size.cc | 37 ++-
> gcc/tree.cc | 133 +++++++++++
> gcc/tree.h | 15 ++
> 14 files changed, 797 insertions(+), 25 deletions(-)
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-05 20:08 ` Siddhesh Poyarekar
@ 2023-10-05 22:35 ` Kees Cook
2023-10-06 5:11 ` Martin Uecker
2023-10-18 21:11 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-05 22:35 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches, uecker,
isanbard
On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
> 2. How would you handle signedness of the size field? The size gets
> converted to sizetype everywhere it is used and overflows/underflows may
> produce interesting results. Do you want to limit the types to unsigned or
> do you want to add a disclaimer in the docs? The former seems like the
> *right* thing to do given that it is a new feature; best to enforce the
> cleaner habit at the outset.
The Linux kernel has a lot of "int" counters, so the goal is to catch
negative offsets just like too-large offsets at runtime with the sanitizer
and report 0 for __bdos. Refactoring all these to be unsigned is going
to take time since at least some of them use the negative values as
special values unrelated to array indexing. :(
So, perhaps if unsigned counters are worth enforcing, can this be a
separate warning the kernel can turn off initially?
-Kees
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-05 22:35 ` Kees Cook
@ 2023-10-06 5:11 ` Martin Uecker
2023-10-06 10:50 ` Siddhesh Poyarekar
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-06 5:11 UTC (permalink / raw)
To: Kees Cook, Siddhesh Poyarekar
Cc: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches, isanbard
Am Donnerstag, dem 05.10.2023 um 15:35 -0700 schrieb Kees Cook:
> On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
> > 2. How would you handle signedness of the size field? The size gets
> > converted to sizetype everywhere it is used and overflows/underflows may
> > produce interesting results. Do you want to limit the types to unsigned or
> > do you want to add a disclaimer in the docs? The former seems like the
> > *right* thing to do given that it is a new feature; best to enforce the
> > cleaner habit at the outset.
>
> The Linux kernel has a lot of "int" counters, so the goal is to catch
> negative offsets just like too-large offsets at runtime with the sanitizer
> and report 0 for __bdos. Refactoring all these to be unsigned is going
> to take time since at least some of them use the negative values as
> special values unrelated to array indexing. :(
>
> So, perhaps if unsigned counters are worth enforcing, can this be a
> separate warning the kernel can turn off initially?
>
I think unsigned counters are much more problematic than signed ones
because wraparound errors are more difficult to find.
With unsigned you could potentially diagnose wraparound, but only if we
add -fsanitize=unsigned-overflow *and* add mechanism to mark intentional
wraparound *and* everybody adds this annotation after carefully screening
their code *and* rewriting all operations such as (counter - 3) + 5
where the wraparound in the intermediate expression is harmless.
For this reason, I do not think we should ever enforce some rule that
the counter has to be unsigned.
What we could do, is detect *storing* negative values into the
counter at run-time using UBSan. (but if negative values are
used for special cases, one also should be able to turn this
off).
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-06 5:11 ` Martin Uecker
@ 2023-10-06 10:50 ` Siddhesh Poyarekar
2023-10-06 20:01 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-06 10:50 UTC (permalink / raw)
To: Martin Uecker, Kees Cook
Cc: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches, isanbard
On 2023-10-06 01:11, Martin Uecker wrote:
> Am Donnerstag, dem 05.10.2023 um 15:35 -0700 schrieb Kees Cook:
>> On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
>>> 2. How would you handle signedness of the size field? The size gets
>>> converted to sizetype everywhere it is used and overflows/underflows may
>>> produce interesting results. Do you want to limit the types to unsigned or
>>> do you want to add a disclaimer in the docs? The former seems like the
>>> *right* thing to do given that it is a new feature; best to enforce the
>>> cleaner habit at the outset.
>>
>> The Linux kernel has a lot of "int" counters, so the goal is to catch
>> negative offsets just like too-large offsets at runtime with the sanitizer
>> and report 0 for __bdos. Refactoring all these to be unsigned is going
>> to take time since at least some of them use the negative values as
>> special values unrelated to array indexing. :(
>>
>> So, perhaps if unsigned counters are worth enforcing, can this be a
>> separate warning the kernel can turn off initially?
>>
>
> I think unsigned counters are much more problematic than signed ones
> because wraparound errors are more difficult to find.
>
> With unsigned you could potentially diagnose wraparound, but only if we
> add -fsanitize=unsigned-overflow *and* add mechanism to mark intentional
> wraparound *and* everybody adds this annotation after carefully screening
> their code *and* rewriting all operations such as (counter - 3) + 5
> where the wraparound in the intermediate expression is harmless.
>
> For this reason, I do not think we should ever enforce some rule that
> the counter has to be unsigned.
>
> What we could do, is detect *storing* negative values into the
> counter at run-time using UBSan. (but if negative values are
> used for special cases, one also should be able to turn this
> off).
All of the object size detection relies on object sizes being sizetype.
The closest we could do with that is detect (sz != SIZE_MAX && sz >
size_t / 2), since allocators typically cannot allocate more than
SIZE_MAX / 2.
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-06 10:50 ` Siddhesh Poyarekar
@ 2023-10-06 20:01 ` Martin Uecker
2023-10-18 15:37 ` Siddhesh Poyarekar
2023-10-18 19:35 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-06 20:01 UTC (permalink / raw)
To: Siddhesh Poyarekar, Kees Cook
Cc: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches, isanbard
Am Freitag, dem 06.10.2023 um 06:50 -0400 schrieb Siddhesh Poyarekar:
> On 2023-10-06 01:11, Martin Uecker wrote:
> > Am Donnerstag, dem 05.10.2023 um 15:35 -0700 schrieb Kees Cook:
> > > On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
> > > > 2. How would you handle signedness of the size field? The size gets
> > > > converted to sizetype everywhere it is used and overflows/underflows may
> > > > produce interesting results. Do you want to limit the types to unsigned or
> > > > do you want to add a disclaimer in the docs? The former seems like the
> > > > *right* thing to do given that it is a new feature; best to enforce the
> > > > cleaner habit at the outset.
> > >
> > > The Linux kernel has a lot of "int" counters, so the goal is to catch
> > > negative offsets just like too-large offsets at runtime with the sanitizer
> > > and report 0 for __bdos. Refactoring all these to be unsigned is going
> > > to take time since at least some of them use the negative values as
> > > special values unrelated to array indexing. :(
> > >
> > > So, perhaps if unsigned counters are worth enforcing, can this be a
> > > separate warning the kernel can turn off initially?
> > >
> >
> > I think unsigned counters are much more problematic than signed ones
> > because wraparound errors are more difficult to find.
> >
> > With unsigned you could potentially diagnose wraparound, but only if we
> > add -fsanitize=unsigned-overflow *and* add mechanism to mark intentional
> > wraparound *and* everybody adds this annotation after carefully screening
> > their code *and* rewriting all operations such as (counter - 3) + 5
> > where the wraparound in the intermediate expression is harmless.
> >
> > For this reason, I do not think we should ever enforce some rule that
> > the counter has to be unsigned.
> >
> > What we could do, is detect *storing* negative values into the
> > counter at run-time using UBSan. (but if negative values are
> > used for special cases, one also should be able to turn this
> > off).
>
> All of the object size detection relies on object sizes being sizetype.
> The closest we could do with that is detect (sz != SIZE_MAX && sz >
> size_t / 2), since allocators typically cannot allocate more than
> SIZE_MAX / 2.
I was talking about the counter in:
struct {
int counter;
char buf[] __counted_by__((counter))
};
which could be checked to be positive either when stored to or
when buf is used.
And yes, we could also check the size of buf. Not sure what is
done for VLAs now, but I guess it could be similar.
Best,
Martin
>
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-10-05 18:51 ` Siddhesh Poyarekar
2023-10-05 19:31 ` Siddhesh Poyarekar
@ 2023-10-18 14:41 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 14:41 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: joseph, richard.guenther, jakub, gcc-patches, keescook, uecker, isanbard
Hi, Sid,
Thanks a lot for your time and effort to review this patch set!
And sorry for my late reply due to a long vacation immediately after Cauldron, just came back this Monday..
See my reply embedded below:
> On Oct 5, 2023, at 2:51 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-08-25 11:24, Qing Zhao wrote:
>> Provide a new counted_by attribute to flexible array member field.
>
> The obligatory "I can't ack the patch but here's a review" disclaimer :)
>
>> 'counted_by (COUNT)'
>> The 'counted_by' attribute may be attached to the flexible array
>> member of a structure. It indicates that the number of the
>> elements of the array is given by the field named "COUNT" in the
>> same structure as the flexible array member. GCC uses this
>> information to improve the results of the array bound sanitizer and
>> the '__builtin_dynamic_object_size'.
>> For instance, the following code:
>> struct P {
>> size_t count;
>> char other;
>> char array[] __attribute__ ((counted_by (count)));
>> } *p;
>> specifies that the 'array' is a flexible array member whose number
>> of elements is given by the field 'count' in the same structure.
>> The field that represents the number of the elements should have an
>> integer type. An explicit 'counted_by' annotation defines a
>> relationship between two objects, 'p->array' and 'p->count', that
>> 'p->array' has _at least_ 'p->count' number of elements available.
>> This relationship must hold even after any of these related objects
>> are updated. It's the user's responsibility to make sure this
>> relationship to be kept all the time. Otherwise the results of the
>> array bound sanitizer and the '__builtin_dynamic_object_size' might
>> be incorrect.
>> For instance, in the following example, the allocated array has
>> less elements than what's specified by the 'sbuf->count', this is
>> an user error. As a result, out-of-bounds access to the array
>> might not be detected.
>> #define SIZE_BUMP 10
>> struct P *sbuf;
>> void alloc_buf (size_t nelems)
>> {
>> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> (offsetof (struct P, array[0])
>> + nelems * sizeof (char))));
>> sbuf->count = nelems + SIZE_BUMP;
>> /* This is invalid when the sbuf->array has less than sbuf->count
>> elements. */
>> }
>> In the following example, the 2nd update to the field 'sbuf->count'
>> of the above structure will permit out-of-bounds access to the
>> array 'sbuf>array' as well.
>> #define SIZE_BUMP 10
>> struct P *sbuf;
>> void alloc_buf (size_t nelems)
>> {
>> sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> (offsetof (struct P, array[0])
>> + (nelems + SIZE_BUMP) * sizeof (char))));
>> sbuf->count = nelems;
>> /* This is valid when the sbuf->array has at least sbuf->count
>> elements. */
>> }
>> void use_buf (int index)
>> {
>> sbuf->count = sbuf->count + SIZE_BUMP + 1;
>> /* Now the value of sbuf->count is larger than the number
>> of elements of sbuf->array. */
>> sbuf->array[index] = 0;
>> /* then the out-of-bound access to this array
>> might not be detected. */
>> }
>> gcc/c-family/ChangeLog:
>> PR C/108896
>> * c-attribs.cc (handle_counted_by_attribute): New function.
>> (attribute_takes_identifier_p): Add counted_by attribute to the list.
>> * c-common.cc (c_flexible_array_member_type_p): ...To this.
>> * c-common.h (c_flexible_array_member_type_p): New prototype.
>> gcc/c/ChangeLog:
>> PR C/108896
>> * c-decl.cc (flexible_array_member_type_p): Renamed and moved to...
>> (add_flexible_array_elts_to_size): Use renamed function.
>> (is_flexible_array_member_p): Use renamed function.
>> (verify_counted_by_attribute): New function.
>> (finish_struct): Use renamed function and verify counted_by
>> attribute.
>> gcc/ChangeLog:
>> PR C/108896
>> * doc/extend.texi: Document attribute counted_by.
>> * tree.cc (get_named_field): New function.
>> * tree.h (get_named_field): New prototype.
>> gcc/testsuite/ChangeLog:
>> PR C/108896
>> * gcc.dg/flex-array-counted-by.c: New test.
>> ---
>> gcc/c-family/c-attribs.cc | 54 ++++++++++++-
>> gcc/c-family/c-common.cc | 13 ++++
>> gcc/c-family/c-common.h | 1 +
>> gcc/c/c-decl.cc | 79 +++++++++++++++-----
>> gcc/doc/extend.texi | 77 +++++++++++++++++++
>> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++++++++
>> gcc/tree.cc | 40 ++++++++++
>> gcc/tree.h | 5 ++
>> 8 files changed, 291 insertions(+), 18 deletions(-)
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> diff --git a/gcc/c-family/c-attribs.cc b/gcc/c-family/c-attribs.cc
>> index e2792ca6898b..65e4f6639109 100644
>> --- a/gcc/c-family/c-attribs.cc
>> +++ b/gcc/c-family/c-attribs.cc
>> @@ -103,6 +103,8 @@ static tree handle_warn_if_not_aligned_attribute (tree *, tree, tree,
>> int, bool *);
>> static tree handle_strict_flex_array_attribute (tree *, tree, tree,
>> int, bool *);
>> +static tree handle_counted_by_attribute (tree *, tree, tree,
>> + int, bool *);
>> static tree handle_weak_attribute (tree *, tree, tree, int, bool *) ;
>> static tree handle_noplt_attribute (tree *, tree, tree, int, bool *) ;
>> static tree handle_alias_ifunc_attribute (bool, tree *, tree, tree, bool *);
>> @@ -373,6 +375,8 @@ const struct attribute_spec c_common_attribute_table[] =
>> handle_warn_if_not_aligned_attribute, NULL },
>> { "strict_flex_array", 1, 1, true, false, false, false,
>> handle_strict_flex_array_attribute, NULL },
>> + { "counted_by", 1, 1, true, false, false, false,
>> + handle_counted_by_attribute, NULL },
>> { "weak", 0, 0, true, false, false, false,
>> handle_weak_attribute, NULL },
>> { "noplt", 0, 0, true, false, false, false,
>> @@ -601,7 +605,8 @@ attribute_takes_identifier_p (const_tree attr_id)
>> else if (!strcmp ("mode", spec->name)
>> || !strcmp ("format", spec->name)
>> || !strcmp ("cleanup", spec->name)
>> - || !strcmp ("access", spec->name))
>> + || !strcmp ("access", spec->name)
>> + || !strcmp ("counted_by", spec->name))
>> return true;
>> else
>> return targetm.attribute_takes_identifier_p (attr_id);
>> @@ -2555,6 +2560,53 @@ handle_strict_flex_array_attribute (tree *node, tree name,
>> return NULL_TREE;
>> }
>> +/* Handle a "counted_by" attribute; arguments as in
>> + struct attribute_spec.handler. */
>> +
>> +static tree
>> +handle_counted_by_attribute (tree *node, tree name,
>> + tree args, int ARG_UNUSED (flags),
>> + bool *no_add_attrs)
>> +{
>> + tree decl = *node;
>> + tree argval = TREE_VALUE (args);
>> +
>> + /* This attribute only applies to field decls of a structure. */
>> + if (TREE_CODE (decl) != FIELD_DECL)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for non-field"
>> + " declaration %q+D", name, decl);
>> + *no_add_attrs = true;
>> + }
>
> Applies only to struct fields. OK.
>
>> + /* This attribute only applies to field with array type. */
>> + else if (TREE_CODE (TREE_TYPE (decl)) != ARRAY_TYPE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for a non-array field",
>> + name);
>> + *no_add_attrs = true;
>> + }
>
> The struct field should also be an array. OK.
>
>> + /* This attribute only applies to a C99 flexible array member type. */
>> + else if (! c_flexible_array_member_type_p (TREE_TYPE (decl)))
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%qE attribute may not be specified for a non"
>> + " flexible array member field",
>> + name);
>> + *no_add_attrs = true;
>> + }
>
> Additionally, the field should be a *flex* array. OK. Could this be reworded to:
>
> %qE attribute only applies to C99 flexible array members
>
> That would make it clear that the GNU extension flex arrays (i.e. any last array member of a struct) don't support this attribute.
Yes, make sense, will update this.
>
>> + /* The argument should be an identifier. */
>> + else if (TREE_CODE (argval) != IDENTIFIER_NODE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (decl),
>> + "%<counted_by%> argument not an identifier");
>> + *no_add_attrs = true;
>> + }
>
> Argument should be an identifier, and the check for validity of the identifier comes later in finish_struct. OK.
>
>> +
>> + return NULL_TREE;
>> +}
>> +
>> /* Handle a "weak" attribute; arguments as in
>> struct attribute_spec.handler. */
>> diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc
>> index 9fbaeb437a12..a18937245c2a 100644
>> --- a/gcc/c-family/c-common.cc
>> +++ b/gcc/c-family/c-common.cc
>> @@ -9521,6 +9521,19 @@ c_common_finalize_early_debug (void)
>> (*debug_hooks->early_global_decl) (cnode->decl);
>> }
>> +/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
>> +bool
>> +c_flexible_array_member_type_p (const_tree type)
>> +{
>> + if (TREE_CODE (type) == ARRAY_TYPE
>> + && TYPE_SIZE (type) == NULL_TREE
>> + && TYPE_DOMAIN (type) != NULL_TREE
>> + && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
>> + return true;
>> +
>> + return false;
>> +}
>> +
>
> Hoist flexible_array_member_type_p to use more widely. OK.
>
>> /* Get the LEVEL of the strict_flex_array for the ARRAY_FIELD based on the
>> values of attribute strict_flex_array and the flag_strict_flex_arrays. */
>> unsigned int
>> diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h
>> index 78fc5248ba68..c29bb429062b 100644
>> --- a/gcc/c-family/c-common.h
>> +++ b/gcc/c-family/c-common.h
>> @@ -909,6 +909,7 @@ extern tree fold_for_warn (tree);
>> extern tree c_common_get_narrower (tree, int *);
>> extern bool get_attribute_operand (tree, unsigned HOST_WIDE_INT *);
>> extern void c_common_finalize_early_debug (void);
>> +extern bool c_flexible_array_member_type_p (const_tree);
>> extern unsigned int c_strict_flex_array_level_of (tree);
>> extern bool c_option_is_from_cpp_diagnostics (int);
>> diff --git a/gcc/c/c-decl.cc b/gcc/c/c-decl.cc
>> index 1f9eb44dbaa2..e943b49b5230 100644
>> --- a/gcc/c/c-decl.cc
>> +++ b/gcc/c/c-decl.cc
>> @@ -5173,19 +5173,6 @@ set_array_declarator_inner (struct c_declarator *decl,
>> return decl;
>> }
>> -/* Determine whether TYPE is a ISO C99 flexible array memeber type "[]". */
>> -static bool
>> -flexible_array_member_type_p (const_tree type)
>> -{
>> - if (TREE_CODE (type) == ARRAY_TYPE
>> - && TYPE_SIZE (type) == NULL_TREE
>> - && TYPE_DOMAIN (type) != NULL_TREE
>> - && TYPE_MAX_VALUE (TYPE_DOMAIN (type)) == NULL_TREE)
>> - return true;
>> -
>> - return false;
>> -}
>> -
>> /* Determine whether TYPE is a one-element array type "[1]". */
>> static bool
>> one_element_array_type_p (const_tree type)
>> @@ -5222,7 +5209,7 @@ add_flexible_array_elts_to_size (tree decl, tree init)
>> elt = CONSTRUCTOR_ELTS (init)->last ().value;
>> type = TREE_TYPE (elt);
>> - if (flexible_array_member_type_p (type))
>> + if (c_flexible_array_member_type_p (type))
>> {
>> complete_array_type (&type, elt, false);
>> DECL_SIZE (decl)
>> @@ -9094,7 +9081,7 @@ is_flexible_array_member_p (bool is_last_field,
>> bool is_zero_length_array = zero_length_array_type_p (TREE_TYPE (x));
>> bool is_one_element_array = one_element_array_type_p (TREE_TYPE (x));
>> - bool is_flexible_array = flexible_array_member_type_p (TREE_TYPE (x));
>> + bool is_flexible_array = c_flexible_array_member_type_p (TREE_TYPE (x));
>> unsigned int strict_flex_array_level = c_strict_flex_array_level_of (x);
>>
>
> Simple refactoring. OK.
>
>> @@ -9124,6 +9111,61 @@ is_flexible_array_member_p (bool is_last_field,
>> return false;
>> }
>> +/* Verify the argument of the counted_by attribute of the flexible array
>
> Verify *that* the argument...
Sure, will update this.
>
>> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
>> + FIELDLIST, Report error and remove this attribute when it's not. */
>> +static void
>> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
>> +{
>> + tree attr_counted_by = lookup_attribute ("counted_by",
>> + DECL_ATTRIBUTES (field_decl));
>> +
>> + if (!attr_counted_by)
>> + return;
>> +
>> + /* If there is an counted_by attribute attached to the field,
>> + verify it. */
>> +
>> + const char *fieldname
>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>> +
>> + /* Verify the argument of the attrbute is a valid field of the
>
> s/attrbute/attribute/
Okay.
>
>> + containing structure. */
>> +
>> + tree counted_by_field = get_named_field (fieldlist, fieldname);
>> +
>> + /* Error when the field is not found in the containing structure. */
>> + if (!counted_by_field)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>> + "%qE attribute argument not a field declaration"
>> + " in the same structure, ignore it",
>> + (get_attribute_name (attr_counted_by)));
>
> Probably someone with English as a first language would make a better suggestion, but how about:
>
> Argument specified in %qE attribute is not a field declaration in the
> same structure, ignoring it.
This is better, will update it.
>
>> +
>> + DECL_ATTRIBUTES (field_decl)
>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>> + }
>> + else
>> + /* Error when the field is not with an integer type. */
>
> Suggest: Flag an error when the field is not of an integer type.
Okay.
>
>> + {
>> + while (TREE_CHAIN (counted_by_field))
>> + counted_by_field = TREE_CHAIN (counted_by_field);
>> + tree real_field = TREE_VALUE (counted_by_field);
>> +
>> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
>> + {
>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>> + "%qE attribute argument not a field declaration"
>> + " with integer type, ignore it",
>> + (get_attribute_name (attr_counted_by)));
>
> Suggest:
>
> Argument specified in %qE attribute is not of an integer type,
> ignoring it.
Will update this.
>
>> +
>> + DECL_ATTRIBUTES (field_decl)
>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>> + }
>> + }
>> +
>> + return;
>> +}
>> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
>> LOC is the location of the RECORD_TYPE or UNION_TYPE's definition.
>> @@ -9244,7 +9286,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
>> DECL_PACKED (x) = 1;
>> /* Detect flexible array member in an invalid context. */
>> - if (flexible_array_member_type_p (TREE_TYPE (x)))
>> + if (c_flexible_array_member_type_p (TREE_TYPE (x)))
>> {
>> if (TREE_CODE (t) == UNION_TYPE)
>> {
>> @@ -9265,6 +9307,9 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
>> "members");
>> TREE_TYPE (x) = error_mark_node;
>> }
>> + /* if there is a counted_by attribute attached to this field,
>> + verify it. */
>> + verify_counted_by_attribute (fieldlist, x);
>> }
>> if (pedantic && TREE_CODE (t) == RECORD_TYPE
>> @@ -9279,7 +9324,7 @@ finish_struct (location_t loc, tree t, tree fieldlist, tree attributes,
>> when x is an array and is the last field. */
>> if (TREE_CODE (TREE_TYPE (x)) == ARRAY_TYPE)
>> TYPE_INCLUDES_FLEXARRAY (t)
>> - = is_last_field && flexible_array_member_type_p (TREE_TYPE (x));
>> + = is_last_field && c_flexible_array_member_type_p (TREE_TYPE (x));
>> /* Recursively set TYPE_INCLUDES_FLEXARRAY for the context of x, t
>> when x is an union or record and is the last field. */
>> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (x)))
>> diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi
>> index 97eaacf8a7ec..ea6240646936 100644
>> --- a/gcc/doc/extend.texi
>> +++ b/gcc/doc/extend.texi
>> @@ -7617,6 +7617,83 @@ When both the attribute and the option present at the same time, the level of
>> the strictness for the specific trailing array field is determined by the
>> attribute.
>> +@cindex @code{counted_by} variable attribute
>> +@item counted_by (@var{count})
>> +The @code{counted_by} attribute may be attached to the flexible array
>> +member of a structure. It indicates that the number of the elements of the
>> +array is given by the field named "@var{count}" in the same structure as the
>> +flexible array member. GCC uses this information to improve the results of
>> +the array bound sanitizer and the @code{__builtin_dynamic_object_size}.
>
> Maybe specify somehow that this only applies to C99 flexible arrays? Like:
>
> The @code{counted_by} attribute may be attached to the C99 flexible
> array member of a structure...
Yes, specify that the attribute will be only applied to C99 FMA is necessary, I will update this.
>
>> +
>> +For instance, the following code:
>> +
>> +@smallexample
>> +struct P @{
>> + size_t count;
>> + char other;
>> + char array[] __attribute__ ((counted_by (count)));
>> +@} *p;
>> +@end smallexample
>> +
>> +@noindent
>> +specifies that the @code{array} is a flexible array member whose number of
>> +elements is given by the field @code{count} in the same structure.
>> +
>> +The field that represents the number of the elements should have an integer
>> +type. An explicit @code{counted_by} annotation defines a relationship between
>> +two objects, @code{p->array} and @code{p->count}, that @code{p->array} has
>> +@emph{at least} @code{p->count} number of elements available. This relationship
>> +must hold even after any of these related objects are updated. It's the user's
>> +responsibility to make sure this relationship to be kept all the time.
>> +Otherwise the results of the array bound sanitizer and the
>> +@code{__builtin_dynamic_object_size} might be incorrect.
>
> Suggest:
>
> The field that represents the number of the elements must have an
> integer type. An explicit @code{counted_by} annotation defines a
> relationship between two objects, @code{p->array} and @code{p->count},
> that @code{p->array} has @emph{at least} @code{p->count} number of
> elements available. The user is responsible to ensure that this
> relationship is consistently maintained during the lifetime of the
> object. Failure to do so may result in results of the array bound
> sanitizer and the @code{__builtin_dynamic_object_size} being
> undefined.
Thanks for the suggestion, will update accordingly.
>
>> +
>> +For instance, in the following example, the allocated array has less elements
>> +than what's specified by the @code{sbuf->count}, this is an user error. As a
>> +result, out-of-bounds access to the array might not be detected.
>> +
>> +@smallexample
>> +#define SIZE_BUMP 10
>> +struct P *sbuf;
>> +void alloc_buf (size_t nelems)
>> +@{
>> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> + (offsetof (struct P, array[0])
>> + + nelems * sizeof (char))));
>> + sbuf->count = nelems + SIZE_BUMP;
>> + /* This is invalid when the sbuf->array has less than sbuf->count
>> + elements. */
>> +@}
>> +@end smallexample
>> +
>> +In the following example, the 2nd update to the field @code{sbuf->count} of
>> +the above structure will permit out-of-bounds access to the array
>> +@code{sbuf>array} as well.
>> +
>> +@smallexample
>> +#define SIZE_BUMP 10
>> +struct P *sbuf;
>> +void alloc_buf (size_t nelems)
>> +@{
>> + sbuf = (struct P *) malloc (MAX (sizeof (struct P),
>> + (offsetof (struct P, array[0])
>> + + (nelems + SIZE_BUMP) * sizeof (char))));
>> + sbuf->count = nelems;
>> + /* This is valid when the sbuf->array has at least sbuf->count
>> + elements. */
>> +@}
>> +void use_buf (int index)
>> +@{
>> + sbuf->count = sbuf->count + SIZE_BUMP + 1;
>> + /* Now the value of sbuf->count is larger than the number
>> + of elements of sbuf->array. */
>> + sbuf->array[index] = 0;
>> + /* then the out-of-bound access to this array
>> + might not be detected. */
>> +@}
>> +@end smallexample
>> +
>> +
>
> I'm unsure if we should be demonstrating example code with undefined behaviour. Maybe the documentation is better off without it.
Good point, will delete the example with undefined behavior from the doc.
>
>> @cindex @code{alloc_size} variable attribute
>> @item alloc_size (@var{position})
>> @itemx alloc_size (@var{position-1}, @var{position-2})
>> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by.c b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> new file mode 100644
>> index 000000000000..f8ce9776bf86
>> --- /dev/null
>> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> @@ -0,0 +1,40 @@
>> +/* testing the correct usage of attribute counted_by. */
>> +/* { dg-do compile } */
>> +/* { dg-options "-O2" } */
>> +
>> +#include <wchar.h>
>> +
>> +int size;
>> +int x __attribute ((counted_by (size))); /* { dg-error "attribute may not be specified for non-field declaration" } */
>> +
>> +struct trailing {
>> + int count;
>> + int field __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non-array field" } */
>> +};
>> +
>> +struct trailing_1 {
>> + int count;
>> + int array_1[0] __attribute ((counted_by (count))); /* { dg-error "attribute may not be specified for a non flexible array member field" } */
>> +};
>> +
>> +int count;
>> +struct trailing_array_2 {
>> + int count;
>> + int array_2[] __attribute ((counted_by ("count"))); /* { dg-error "argument not an identifier" } */
>> +};
>> +
>> +struct trailing_array_3 {
>> + int other;
>> + int array_3[] __attribute ((counted_by (L"count"))); /* { dg-error "argument not an identifier" } */
>> +};
>> +
>> +struct trailing_array_4 {
>> + int other;
>> + int array_4[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration in the same structure, ignore it" } */
>> +};
>> +
>> +int count;
>> +struct trailing_array_5 {
>> + float count;
>> + int array_5[] __attribute ((counted_by (count))); /* { dg-error "attribute argument not a field declaration with integer type, ignore it" } */
>> +};
>
> Tests look OK in principle, but may need updating for the error message.
Okay.
thanks.
Qing
>
>> diff --git a/gcc/tree.cc b/gcc/tree.cc
>> index 420857b110c4..fcd36ae0cd74 100644
>> --- a/gcc/tree.cc
>> +++ b/gcc/tree.cc
>> @@ -12745,6 +12745,46 @@ array_ref_element_size (tree exp)
>> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
>> }
>> +/* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
>> + with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
>> + whose name is FIELDNAME, which is the last TREE_VALUE of the list.
>> + return NULL_TREE if such field is not found. Normally this list is of
>> + length one, but if the field is embedded with (nested) anonymous structures
>> + or unions, this list steps down the chain to the field. */
>> +tree
>> +get_named_field (tree fieldlist, const char *fieldname)
>> +{
>> + tree named_field = NULL_TREE;
>> + for (tree field = fieldlist; field; field = DECL_CHAIN (field))
>> + {
>> + if (TREE_CODE (field) != FIELD_DECL)
>> + continue;
>> + if (DECL_NAME (field) != NULL)
>> + if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), fieldname) == 0)
>> + {
>> + named_field = tree_cons (NULL_TREE, field, named_field);
>> + break;
>> + }
>> + else
>> + continue;
>> + /* if the field is an anonymous struct/union, we will check the nested
>> + fields inside it recursively. */
>> + else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
>> + if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
>> + fieldname)) != NULL_TREE)
>> + {
>> + named_field = tree_cons (NULL_TREE, field, named_field);
>> + break;
>> + }
>> + else
>> + continue;
>> + else
>> + continue;
>> + }
>> + return named_field;
>> +}
>
> Descending recursively into the field list of a struct to find an identifier with a matching name. OK.
>
>> +
>> +
>> /* Return a tree representing the lower bound of the array mentioned in
>> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> diff --git a/gcc/tree.h b/gcc/tree.h
>> index 4c04245e2b1b..4859becaa1e7 100644
>> --- a/gcc/tree.h
>> +++ b/gcc/tree.h
>> @@ -5619,6 +5619,11 @@ extern tree get_base_address (tree t);
>> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> extern tree array_ref_element_size (tree);
>> +/* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
>> + name is FIELDNAME, return NULL_TREE if such field is not found.
>> + searching nested anonymous structure/union recursively. */
>> +extern tree get_named_field (tree, const char *);
>> +
>> /* Return a typenode for the "standard" C type with a given name. */
>> extern tree get_typenode_from_name (const char *);
>>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-10-05 19:31 ` Siddhesh Poyarekar
@ 2023-10-18 14:51 ` Qing Zhao
2023-10-18 15:18 ` Siddhesh Poyarekar
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 14:51 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: joseph, richard.guenther, jakub, gcc-patches, keescook, uecker, isanbard
>>> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
>>> + FIELDLIST, Report error and remove this attribute when it's not. */
>>> +static void
>>> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
>>> +{
>>> + tree attr_counted_by = lookup_attribute ("counted_by",
>>> + DECL_ATTRIBUTES (field_decl));
>>> +
>>> + if (!attr_counted_by)
>>> + return;
>>> +
>>> + /* If there is an counted_by attribute attached to the field,
>>> + verify it. */
>>> +
>>> + const char *fieldname
>>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>>> +
>>> + /* Verify the argument of the attrbute is a valid field of the
>> s/attrbute/attribute/
>>> + containing structure. */
>>> +
>>> + tree counted_by_field = get_named_field (fieldlist, fieldname);
>>> +
>>> + /* Error when the field is not found in the containing structure. */
>>> + if (!counted_by_field)
>>> + {
>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>> + "%qE attribute argument not a field declaration"
>>> + " in the same structure, ignore it",
>>> + (get_attribute_name (attr_counted_by)));
>> Probably someone with English as a first language would make a better suggestion, but how about:
>> Argument specified in %qE attribute is not a field declaration in the
>> same structure, ignoring it.
>>> +
>>> + DECL_ATTRIBUTES (field_decl)
>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>> + }
>>> + else
>>> + /* Error when the field is not with an integer type. */
>> Suggest: Flag an error when the field is not of an integer type.
>>> + {
>>> + while (TREE_CHAIN (counted_by_field))
>>> + counted_by_field = TREE_CHAIN (counted_by_field);
>>> + tree real_field = TREE_VALUE (counted_by_field);
>>> +
>>> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
>>> + {
>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>> + "%qE attribute argument not a field declaration"
>>> + " with integer type, ignore it",
>>> + (get_attribute_name (attr_counted_by)));
>> Suggest:
>> Argument specified in %qE attribute is not of an integer type,
>> ignoring it.
>>> +
>>> + DECL_ATTRIBUTES (field_decl)
>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>> + }
>>> + }
>>> +
>>> + return;
>
> I forgot to mention the redundant return here.
Could you please clarify a little bit here, why the return here is redundant?
>
>>> +}
>>> /* Fill in the fields of a RECORD_TYPE or UNION_TYPE node, T.
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-10-18 14:51 ` Qing Zhao
@ 2023-10-18 15:18 ` Siddhesh Poyarekar
2023-10-18 15:37 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-18 15:18 UTC (permalink / raw)
To: Qing Zhao
Cc: joseph, richard.guenther, jakub, gcc-patches, keescook, uecker, isanbard
On 2023-10-18 10:51, Qing Zhao wrote:
>
>>>> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
>>>> + FIELDLIST, Report error and remove this attribute when it's not. */
>>>> +static void
>>>> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
>>>> +{
>>>> + tree attr_counted_by = lookup_attribute ("counted_by",
>>>> + DECL_ATTRIBUTES (field_decl));
>>>> +
>>>> + if (!attr_counted_by)
>>>> + return;
>>>> +
>>>> + /* If there is an counted_by attribute attached to the field,
>>>> + verify it. */
>>>> +
>>>> + const char *fieldname
>>>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>>>> +
>>>> + /* Verify the argument of the attrbute is a valid field of the
>>> s/attrbute/attribute/
>>>> + containing structure. */
>>>> +
>>>> + tree counted_by_field = get_named_field (fieldlist, fieldname);
>>>> +
>>>> + /* Error when the field is not found in the containing structure. */
>>>> + if (!counted_by_field)
>>>> + {
>>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>>> + "%qE attribute argument not a field declaration"
>>>> + " in the same structure, ignore it",
>>>> + (get_attribute_name (attr_counted_by)));
>>> Probably someone with English as a first language would make a better suggestion, but how about:
>>> Argument specified in %qE attribute is not a field declaration in the
>>> same structure, ignoring it.
>>>> +
>>>> + DECL_ATTRIBUTES (field_decl)
>>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>>> + }
>>>> + else
>>>> + /* Error when the field is not with an integer type. */
>>> Suggest: Flag an error when the field is not of an integer type.
>>>> + {
>>>> + while (TREE_CHAIN (counted_by_field))
>>>> + counted_by_field = TREE_CHAIN (counted_by_field);
>>>> + tree real_field = TREE_VALUE (counted_by_field);
>>>> +
>>>> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
>>>> + {
>>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>>> + "%qE attribute argument not a field declaration"
>>>> + " with integer type, ignore it",
>>>> + (get_attribute_name (attr_counted_by)));
>>> Suggest:
>>> Argument specified in %qE attribute is not of an integer type,
>>> ignoring it.
>>>> +
>>>> + DECL_ATTRIBUTES (field_decl)
>>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>>> + }
>>>> + }
>>>> +
>>>> + return;
>>
>> I forgot to mention the redundant return here.
>
> Could you please clarify a little bit here, why the return here is redundant?
It's the last line in the function, so even without that statement the
function will return.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 1/3] Provide counted_by attribute to flexible array member field (PR108896)
2023-10-18 15:18 ` Siddhesh Poyarekar
@ 2023-10-18 15:37 ` Qing Zhao
0 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 15:37 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: joseph, richard.guenther, jakub, gcc-patches, keescook, uecker, isanbard
> On Oct 18, 2023, at 11:18 AM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-18 10:51, Qing Zhao wrote:
>>>>> + member FIELD_DECL is a valid field of the containing structure's fieldlist,
>>>>> + FIELDLIST, Report error and remove this attribute when it's not. */
>>>>> +static void
>>>>> +verify_counted_by_attribute (tree fieldlist, tree field_decl)
>>>>> +{
>>>>> + tree attr_counted_by = lookup_attribute ("counted_by",
>>>>> + DECL_ATTRIBUTES (field_decl));
>>>>> +
>>>>> + if (!attr_counted_by)
>>>>> + return;
>>>>> +
>>>>> + /* If there is an counted_by attribute attached to the field,
>>>>> + verify it. */
>>>>> +
>>>>> + const char *fieldname
>>>>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>>>>> +
>>>>> + /* Verify the argument of the attrbute is a valid field of the
>>>> s/attrbute/attribute/
>>>>> + containing structure. */
>>>>> +
>>>>> + tree counted_by_field = get_named_field (fieldlist, fieldname);
>>>>> +
>>>>> + /* Error when the field is not found in the containing structure. */
>>>>> + if (!counted_by_field)
>>>>> + {
>>>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>>>> + "%qE attribute argument not a field declaration"
>>>>> + " in the same structure, ignore it",
>>>>> + (get_attribute_name (attr_counted_by)));
>>>> Probably someone with English as a first language would make a better suggestion, but how about:
>>>> Argument specified in %qE attribute is not a field declaration in the
>>>> same structure, ignoring it.
>>>>> +
>>>>> + DECL_ATTRIBUTES (field_decl)
>>>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>>>> + }
>>>>> + else
>>>>> + /* Error when the field is not with an integer type. */
>>>> Suggest: Flag an error when the field is not of an integer type.
>>>>> + {
>>>>> + while (TREE_CHAIN (counted_by_field))
>>>>> + counted_by_field = TREE_CHAIN (counted_by_field);
>>>>> + tree real_field = TREE_VALUE (counted_by_field);
>>>>> +
>>>>> + if (TREE_CODE (TREE_TYPE (real_field)) != INTEGER_TYPE)
>>>>> + {
>>>>> + error_at (DECL_SOURCE_LOCATION (field_decl),
>>>>> + "%qE attribute argument not a field declaration"
>>>>> + " with integer type, ignore it",
>>>>> + (get_attribute_name (attr_counted_by)));
>>>> Suggest:
>>>> Argument specified in %qE attribute is not of an integer type,
>>>> ignoring it.
>>>>> +
>>>>> + DECL_ATTRIBUTES (field_decl)
>>>>> + = remove_attribute ("counted_by", DECL_ATTRIBUTES (field_decl));
>>>>> + }
>>>>> + }
>>>>> +
>>>>> + return;
>>>
>>> I forgot to mention the redundant return here.
>> Could you please clarify a little bit here, why the return here is redundant?
>
> It's the last line in the function, so even without that statement the function will return.
Oh, I see. -:)
Actually,I always put an explicit return there even though it’s the last line and return implicitly.
Qing
>
> Thanks,
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-06 20:01 ` Martin Uecker
@ 2023-10-18 15:37 ` Siddhesh Poyarekar
2023-10-18 19:35 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-18 15:37 UTC (permalink / raw)
To: Martin Uecker, Kees Cook
Cc: Qing Zhao, joseph, richard.guenther, jakub, gcc-patches, isanbard
[Sorry, I forgot to respond to this]
On 2023-10-06 16:01, Martin Uecker wrote:
> Am Freitag, dem 06.10.2023 um 06:50 -0400 schrieb Siddhesh Poyarekar:
>> On 2023-10-06 01:11, Martin Uecker wrote:
>>> Am Donnerstag, dem 05.10.2023 um 15:35 -0700 schrieb Kees Cook:
>>>> On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
>>>>> 2. How would you handle signedness of the size field? The size gets
>>>>> converted to sizetype everywhere it is used and overflows/underflows may
>>>>> produce interesting results. Do you want to limit the types to unsigned or
>>>>> do you want to add a disclaimer in the docs? The former seems like the
>>>>> *right* thing to do given that it is a new feature; best to enforce the
>>>>> cleaner habit at the outset.
>>>>
>>>> The Linux kernel has a lot of "int" counters, so the goal is to catch
>>>> negative offsets just like too-large offsets at runtime with the sanitizer
>>>> and report 0 for __bdos. Refactoring all these to be unsigned is going
>>>> to take time since at least some of them use the negative values as
>>>> special values unrelated to array indexing. :(
>>>>
>>>> So, perhaps if unsigned counters are worth enforcing, can this be a
>>>> separate warning the kernel can turn off initially?
>>>>
>>>
>>> I think unsigned counters are much more problematic than signed ones
>>> because wraparound errors are more difficult to find.
>>>
>>> With unsigned you could potentially diagnose wraparound, but only if we
>>> add -fsanitize=unsigned-overflow *and* add mechanism to mark intentional
>>> wraparound *and* everybody adds this annotation after carefully screening
>>> their code *and* rewriting all operations such as (counter - 3) + 5
>>> where the wraparound in the intermediate expression is harmless.
>>>
>>> For this reason, I do not think we should ever enforce some rule that
>>> the counter has to be unsigned.
>>>
>>> What we could do, is detect *storing* negative values into the
>>> counter at run-time using UBSan. (but if negative values are
>>> used for special cases, one also should be able to turn this
>>> off).
>>
>> All of the object size detection relies on object sizes being sizetype.
>> The closest we could do with that is detect (sz != SIZE_MAX && sz >
>> size_t / 2), since allocators typically cannot allocate more than
>> SIZE_MAX / 2.
>
> I was talking about the counter in:
>
> struct {
> int counter;
> char buf[] __counted_by__((counter))
> };
>
> which could be checked to be positive either when stored to or
> when buf is used.
>
> And yes, we could also check the size of buf. Not sure what is
> done for VLAs now, but I guess it could be similar.
Right now all object sizes are cast to sizetype and the generated
dynamic expressions are such that overflows will result in the computed
object size being zero. Non-generated expressions (like we could get
with __counted_by__) will simply be cast; there's probably scope for
improvement here, where we wrap that with an expression that returns 0
if the size exceeds SIZE_MAX / 2 since that's typically the limit for
allocators. We use that heuristic elsewhere in the __bos/__bdos logic too.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-06 20:01 ` Martin Uecker
2023-10-18 15:37 ` Siddhesh Poyarekar
@ 2023-10-18 19:35 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 19:35 UTC (permalink / raw)
To: Martin Uecker
Cc: Siddhesh Poyarekar, Kees Cook, joseph, richard.guenther, jakub,
gcc-patches, isanbard
> On Oct 6, 2023, at 4:01 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Freitag, dem 06.10.2023 um 06:50 -0400 schrieb Siddhesh Poyarekar:
>> On 2023-10-06 01:11, Martin Uecker wrote:
>>> Am Donnerstag, dem 05.10.2023 um 15:35 -0700 schrieb Kees Cook:
>>>> On Thu, Oct 05, 2023 at 04:08:52PM -0400, Siddhesh Poyarekar wrote:
>>>>> 2. How would you handle signedness of the size field? The size gets
>>>>> converted to sizetype everywhere it is used and overflows/underflows may
>>>>> produce interesting results. Do you want to limit the types to unsigned or
>>>>> do you want to add a disclaimer in the docs? The former seems like the
>>>>> *right* thing to do given that it is a new feature; best to enforce the
>>>>> cleaner habit at the outset.
>>>>
>>>> The Linux kernel has a lot of "int" counters, so the goal is to catch
>>>> negative offsets just like too-large offsets at runtime with the sanitizer
>>>> and report 0 for __bdos. Refactoring all these to be unsigned is going
>>>> to take time since at least some of them use the negative values as
>>>> special values unrelated to array indexing. :(
>>>>
>>>> So, perhaps if unsigned counters are worth enforcing, can this be a
>>>> separate warning the kernel can turn off initially?
>>>>
>>>
>>> I think unsigned counters are much more problematic than signed ones
>>> because wraparound errors are more difficult to find.
>>>
>>> With unsigned you could potentially diagnose wraparound, but only if we
>>> add -fsanitize=unsigned-overflow *and* add mechanism to mark intentional
>>> wraparound *and* everybody adds this annotation after carefully screening
>>> their code *and* rewriting all operations such as (counter - 3) + 5
>>> where the wraparound in the intermediate expression is harmless.
>>>
>>> For this reason, I do not think we should ever enforce some rule that
>>> the counter has to be unsigned.
>>>
>>> What we could do, is detect *storing* negative values into the
>>> counter at run-time using UBSan. (but if negative values are
>>> used for special cases, one also should be able to turn this
>>> off).
>>
>> All of the object size detection relies on object sizes being sizetype.
>> The closest we could do with that is detect (sz != SIZE_MAX && sz >
>> size_t / 2), since allocators typically cannot allocate more than
>> SIZE_MAX / 2.
>
> I was talking about the counter in:
>
> struct {
> int counter;
> char buf[] __counted_by__((counter))
> };
>
> which could be checked to be positive either when stored to or
> when buf is used.
>
> And yes, we could also check the size of buf. Not sure what is
> done for VLAs now, but I guess it could be similar.
>
For VLAs, the bounds expression could be both signed or unsigned.
But we have added a sanitizer option -fsanitize=vla-bound to catch the cases when the size of the VLA is not positive.
For example:
opc@qinzhao-ol8u3-x86 Martin]$ cat t3.c
#include <stdio.h>
size_t foo(int m)
{
char t[m];
return sizeof(t);
}
int main()
{
printf ("the sizeof flexm is %lu \n", foo(-100000000));
return 0;
}
[opc@qinzhao-ol8u3-x86 Martin]$ sh t
/home/opc/Install/latest-d/bin/gcc -fsanitize=undefined -O2 -Wall -Wpedantic t3.c
t3.c:4:8: runtime error: variable length array bound evaluates to non-positive value -100000000
the sizeof flexm is 18446744073609551616
We can do the same thing for “counted_by”. i.e:
1. No specification for signed or unsigned for counted_by field.
2. Add an sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
Is this good enough?
Qing
> Best,
> Martin
>
>
>
>>
>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 2/3] Use the counted_by atribute info in builtin object size [PR108896]
2023-10-05 20:01 ` Siddhesh Poyarekar
@ 2023-10-18 20:39 ` Qing Zhao
0 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 20:39 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: joseph, richard.guenther, jakub, gcc-patches, keescook, uecker, isanbard
Hi, Sid,
Thanks a lot for the detailed comments.
See my responds embedded below.
Qing
> On Oct 5, 2023, at 4:01 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
>
>
> On 2023-08-25 11:24, Qing Zhao wrote:
>> Use the counted_by atribute info in builtin object size to compute the
>> subobject size for flexible array members.
>> gcc/ChangeLog:
>> PR C/108896
>> * tree-object-size.cc (addr_object_size): Use the counted_by
>> attribute info.
>> * tree.cc (component_ref_has_counted_by_p): New function.
>> (component_ref_get_counted_by): New function.
>> * tree.h (component_ref_has_counted_by_p): New prototype.
>> (component_ref_get_counted_by): New prototype.
>> gcc/testsuite/ChangeLog:
>> PR C/108896
>> * gcc.dg/flex-array-counted-by-2.c: New test.
>> * gcc.dg/flex-array-counted-by-3.c: New test.
>> ---
>> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
>> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
>> gcc/tree-object-size.cc | 37 ++-
>> gcc/tree.cc | 95 +++++++-
>> gcc/tree.h | 10 +
>> 5 files changed, 418 insertions(+), 8 deletions(-)
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
>> new file mode 100644
>> index 000000000000..ec580c1f1f01
>> --- /dev/null
>> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
>> @@ -0,0 +1,74 @@
>> +/* test the attribute counted_by and its usage in
>> + * __builtin_dynamic_object_size. */
>> +/* { dg-do run } */
>> +/* { dg-options "-O2" } */
>> +
>> +#include "builtin-object-size-common.h"
>> +
>> +#define expect(p, _v) do { \
>> + size_t v = _v; \
>> + if (p == v) \
>> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
>> + else \
>> + { \
>> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
>> + FAIL (); \
>> + } \
>> +} while (0);
>
> You're using this in a bunch of tests already; does it make sense to consolidate it into builtin-object-size-common.h?
Will do this.
>
>> +
>> +struct flex {
>> + int b;
>> + int c[];
>> +} *array_flex;
>> +
>> +struct annotated {
>> + int b;
>> + int c[] __attribute__ ((counted_by (b)));
>> +} *array_annotated;
>> +
>> +struct nested_annotated {
>> + struct {
>> + union {
>> + int b;
>> + float f;
>> + };
>> + int n;
>> + };
>> + int c[] __attribute__ ((counted_by (b)));
>> +} *array_nested_annotated;
>> +
>> +void __attribute__((__noinline__)) setup (int normal_count, int attr_count)
>> +{
>> + array_flex
>> + = (struct flex *)malloc (sizeof (struct flex)
>> + + normal_count * sizeof (int));
>> + array_flex->b = normal_count;
>> +
>> + array_annotated
>> + = (struct annotated *)malloc (sizeof (struct annotated)
>> + + attr_count * sizeof (int));
>> + array_annotated->b = attr_count;
>> +
>> + array_nested_annotated
>> + = (struct nested_annotated *)malloc (sizeof (struct nested_annotated)
>> + + attr_count * sizeof (int));
>> + array_nested_annotated->b = attr_count;
>> +
>> + return;
>> +}
>> +
>> +void __attribute__((__noinline__)) test ()
>> +{
>> + expect(__builtin_dynamic_object_size(array_flex->c, 1), -1);
>> + expect(__builtin_dynamic_object_size(array_annotated->c, 1),
>> + array_annotated->b * sizeof (int));
>> + expect(__builtin_dynamic_object_size(array_nested_annotated->c, 1),
>> + array_nested_annotated->b * sizeof (int));
>> +}
>
> Maybe another test where the allocation, size assignment and __bdos call happen in the same function, where the allocator is not recognized by gcc:
>
> void *
> __attribute__ ((noinline))
> alloc (size_t sz)
> {
> return __builtin_malloc (sz);
> }
>
> void test (size_t sz)
> {
> array_annotated = alloc (sz);
> array_annotated->b = sz;
> return __builtin_dynamic_object_size (array_annotated->c, 1);
> }
>
> The interesting thing to test (and ensure in the codegen) is that the assignment to array_annotated->b does not get reordered to below the __builtin_dynamic_object_size call since technically there is no data dependency between the two.
Good point.
Will add such testing case.
>
>> +
>> +int main(int argc, char *argv[])
>> +{
>> + setup (10,10);
>> + test ();
>> + DONE ();
>> +}
>> diff --git a/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>> new file mode 100644
>> index 000000000000..a0c3cb88ec71
>> --- /dev/null
>> +++ b/gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>> @@ -0,0 +1,210 @@
>> +/* test the attribute counted_by and its usage in
>> +__builtin_dynamic_object_size: what's the correct behavior when the
>> +allocation size mismatched with the value of counted_by attribute? */
>
> If the behaviour is undefined, does it make sense to add tests for this? Maybe once you have a -Wmismatched-counted-by or similar, we could have tests for that. I guess the counter-argument is that we keep track of this behaviour but not necessarily guarantee it.
This testing case was added mainly for documentation purpose. It includes a detailed explanation on how the current _bdo estimates the size of the object.
Even though there is mismatch between the actual allocation size and the value of counted_by attribute, the behavior of the compiler is still defined based on the algorithm.
When -Wmismatched-counted-by is added later, we can update this testing case with new warning messages, but the behavior of _bdo still keep the same.
>
>> +/* { dg-do run } */
>> +/* { dg-options "-O -fstrict-flex-arrays=3" } */
>> +
>> +#include "builtin-object-size-common.h"
>> +
>> +struct annotated {
>> + size_t foo;
>> + char others;
>> + char array[] __attribute__((counted_by (foo)));
>> +};
>> +
>> +#define expect(p, _v) do { \
>> + size_t v = _v; \
>> + if (p == v) \
>> + __builtin_printf ("ok: %s == %zd\n", #p, p); \
>> + else \
>> + { \
>> + __builtin_printf ("WAT: %s == %zd (expected %zd)\n", #p, p, v); \
>> + FAIL (); \
>> + } \
>> +} while (0);
>
> Same, maybe consolidate this into builtin-object-size-common.h.
Okay.
>
>> +
>> +#define noinline __attribute__((__noinline__))
>> +#define SIZE_BUMP 10
>> +#define MAX(a, b) ((a) > (b) ? (a) : (b))
>> +#define MIN(a, b) ((a) < (b) ? (a) : (b))
>> +
>> +/* In general, Due to type casting, the type for the pointee of a pointer
>> + does not say anything about the object it points to,
>> + So, __builtin_object_size can not directly use the type of the pointee
>> + to decide the size of the object the pointer points to.
>> +
>> + there are only two reliable ways:
>> + A. observed allocations (call to the allocation functions in the routine)
>> + B. observed accesses (read or write access to the location of the
>> + pointer points to)
>> +
>> + that provide information about the type/existence of an object at
>> + the corresponding address.
>> +
>> + for A, we use the "alloc_size" attribute for the corresponding allocation
>> + functions to determine the object size;
>> +
>> + For B, we use the SIZE info of the TYPE attached to the corresponding access.
>> + (We treat counted_by attribute as a complement to the SIZE info of the TYPE
>> + for FMA)
>> +
>> + The only other way in C which ensures that a pointer actually points
>> + to an object of the correct type is 'static':
>> +
>> + void foo(struct P *p[static 1]);
>> +
>> + See https://gcc.gnu.org/pipermail/gcc-patches/2023-July/624814.html
>> + for more details. */
>> +
>> +/* in the following function, malloc allocated more space than the value
>> + of counted_by attribute. Then what's the correct behavior we expect
>> + the __builtin_dynamic_object_size should have for each of the cases? */
>> +
>> +static struct annotated * noinline alloc_buf_more (size_t index)
>> +{
>> + struct annotated *p;
>> + size_t allocated_size
>> + = MAX (sizeof (struct annotated),
>> + (__builtin_offsetof (struct annotated, array[0])
>> + + (index + SIZE_BUMP) * sizeof (char)));
>> + p = (struct annotated *) malloc (allocated_size);
>> +
>> + p->foo = index;
>> +
>> + /*when checking the observed access p->array, we have info on both
>> + observered allocation and observed access,
>> + A. from observed allocation:
>> + allocated_size - offsetof (struct annotated, array[0])
>> + B. from observed access: p->foo * sizeof (char)
>> + */
>> +
>> + /* for size in the whole object: always uses A. */
>> + /* for size in the sub-object: chose the smaller of A and B.
>> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
>> + * for details on why. */
>> +
>> + /* for MAXIMUM size in the whole object: use the allocation size
>> + for the whole object. */
>> + expect(__builtin_dynamic_object_size(p->array, 0),
>> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
>> +
>> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
>> + expect(__builtin_dynamic_object_size(p->array, 1),
>> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
>> + (p->foo) * sizeof(char)));
>> +
>> + /* for MINIMUM size in the whole object: use the allocation size
>> + for the whole object. */
>> + expect(__builtin_dynamic_object_size(p->array, 2),
>> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
>> +
>> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
>> + expect(__builtin_dynamic_object_size(p->array, 3),
>> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
>> + (p->foo) * sizeof(char)));
>> +
>> + /*when checking the pointer p, we only have info on the observed allocation.
>> + So, the object size info can only been obtained from the call to malloc.
>> + for both MAXIMUM and MINIMUM: A = (index + SIZE_BUMP) * sizeof (char) */
>> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
>> + return p;
>> +}
>> +
>> +/* in the following function, malloc allocated less space than the value
>> + of counted_by attribute. Then what's the correct behavior we expect
>> + the __builtin_dynamic_object_size should have for each of the cases?
>> + NOTE: this is an user error, GCC should issue warnings for such case.
>> + this is a seperate issue we should address later. */
>> +
>> +static struct annotated * noinline alloc_buf_less (size_t index)
>> +{
>> + struct annotated *p;
>> + size_t allocated_size
>> + = MAX (sizeof (struct annotated),
>> + (__builtin_offsetof (struct annotated, array[0])
>> + + (index) * sizeof (char)));
>> + p = (struct annotated *) malloc (allocated_size);
>> +
>> + p->foo = index + SIZE_BUMP;
>> +
>> + /*when checking the observed access p->array, we have info on both
>> + observered allocation and observed access,
>> + A. from observed allocation:
>> + allocated_size - offsetof (struct annotated, array[0])
>> + B. from observed access: p->foo * sizeof (char)
>> + */
>> +
>> + /* for size in the whole object: always uses A. */
>> + /* for size in the sub-object: chose the smaller of A and B.
>> + * Please see https://gcc.gnu.org/pipermail/gcc-patches/2023-July/625891.html
>> + * for details on why. */
>> +
>> + /* for MAXIMUM size in the whole object: use the allocation size
>> + for the whole object. */
>> + expect(__builtin_dynamic_object_size(p->array, 0),
>> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
>> +
>> + /* for MAXIMUM size in the sub-object. use the smaller of A and B. */
>> + expect(__builtin_dynamic_object_size(p->array, 1),
>> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
>> + (p->foo) * sizeof(char)));
>> +
>> + /* for MINIMUM size in the whole object: use the allocation size
>> + for the whole object. */
>> + expect(__builtin_dynamic_object_size(p->array, 2),
>> + allocated_size - __builtin_offsetof (struct annotated, array[0]));
>> +
>> + /* for MINIMUM size in the sub-object: use the smaller of A and B. */
>> + expect(__builtin_dynamic_object_size(p->array, 3),
>> + MIN (allocated_size - __builtin_offsetof (struct annotated, array[0]),
>> + (p->foo) * sizeof(char)));
>> +
>> + /*when checking the pointer p, we only have info on the observed
>> + allocation. So, the object size info can only been obtained from
>> + the call to malloc. */
>> + expect(__builtin_dynamic_object_size(p, 0), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 1), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 2), allocated_size);
>> + expect(__builtin_dynamic_object_size(p, 3), allocated_size);
>> + return p;
>> +}
>> +
>> +int main ()
>> +{
>> + struct annotated *p, *q;
>> + p = alloc_buf_more (10);
>> + q = alloc_buf_less (10);
>> +
>> + /*when checking the observed access p->array, we only have info on the
>> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
>> + info on the whole object. */
>> + expect(__builtin_dynamic_object_size(p->array, 0), -1);
>> + expect(__builtin_dynamic_object_size(p->array, 1), p->foo * sizeof(char));
>> + expect(__builtin_dynamic_object_size(p->array, 2), 0);
>> + expect(__builtin_dynamic_object_size(p->array, 3), p->foo * sizeof(char));
>> + /*when checking the pointer p, we have no observed allocation nor observed
>> + access, therefore, we cannot determine the size info here. */
>> + expect(__builtin_dynamic_object_size(p, 0), -1);
>> + expect(__builtin_dynamic_object_size(p, 1), -1);
>> + expect(__builtin_dynamic_object_size(p, 2), 0);
>> + expect(__builtin_dynamic_object_size(p, 3), 0);
>> +
>> + /*when checking the observed access p->array, we only have info on the
>> + observed access, i.e, the TYPE_SIZE info from the access. We don't have
>> + info on the whole object. */
>> + expect(__builtin_dynamic_object_size(q->array, 0), -1);
>> + expect(__builtin_dynamic_object_size(q->array, 1), q->foo * sizeof(char));
>> + expect(__builtin_dynamic_object_size(q->array, 2), 0);
>> + expect(__builtin_dynamic_object_size(q->array, 3), q->foo * sizeof(char));
>> + /*when checking the pointer p, we have no observed allocation nor observed
>> + access, therefore, we cannot determine the size info here. */
>> + expect(__builtin_dynamic_object_size(q, 0), -1);
>> + expect(__builtin_dynamic_object_size(q, 1), -1);
>> + expect(__builtin_dynamic_object_size(q, 2), 0);
>> + expect(__builtin_dynamic_object_size(q, 3), 0);
>> +
>> + DONE ();
>> +}
>> diff --git a/gcc/tree-object-size.cc b/gcc/tree-object-size.cc
>> index a62af0500563..cf7843c5684b 100644
>> --- a/gcc/tree-object-size.cc
>> +++ b/gcc/tree-object-size.cc
>> @@ -585,6 +585,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> if (pt_var != TREE_OPERAND (ptr, 0))
>> {
>> tree var;
>> + tree counted_by_ref = NULL_TREE;
>> if (object_size_type & OST_SUBOBJECT)
>> {
>> @@ -600,11 +601,12 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> var = TREE_OPERAND (var, 0);
>> if (var != pt_var && TREE_CODE (var) == ARRAY_REF)
>> var = TREE_OPERAND (var, 0);
>> - if (! TYPE_SIZE_UNIT (TREE_TYPE (var))
>> + if (! component_ref_has_counted_by_p (var)
>> + && ((! TYPE_SIZE_UNIT (TREE_TYPE (var))
>> || ! tree_fits_uhwi_p (TYPE_SIZE_UNIT (TREE_TYPE (var)))
>> || (pt_var_size && TREE_CODE (pt_var_size) == INTEGER_CST
>> && tree_int_cst_lt (pt_var_size,
>> - TYPE_SIZE_UNIT (TREE_TYPE (var)))))
>> + TYPE_SIZE_UNIT (TREE_TYPE (var)))))))
>> var = pt_var;
>> else if (var != pt_var && TREE_CODE (pt_var) == MEM_REF)
>> {
>
> Hmm, only for subobject size? I thought we had consensus on using sizeof (struct) + counted_by_size as the conservative maximum size for whole object size too, didn't we?
Yes, in this initial patch set, only minimum change to tree-object-size.cc. therefore only handle subobject size. (And subobject size is more important for linux kernel security purpose)
And I will add a follow up patch to add new code into tree-object-size.cc to support whole object size by using sizeof(struct) + counted_by_size.
Is this Okay?
>
>> @@ -612,6 +614,7 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> /* For &X->fld, compute object size if fld isn't a flexible array
>> member. */
>> bool is_flexible_array_mem_ref = false;
>> +
>> while (v && v != pt_var)
>> switch (TREE_CODE (v))
>> {
>
> Unnecessary newline.
Okay.
>
>> @@ -660,6 +663,8 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> /* Now the ref is to an array type. */
>> gcc_assert (TREE_CODE (TREE_TYPE (v)) == ARRAY_TYPE);
>> is_flexible_array_mem_ref = array_ref_flexible_size_p (v);
>> + counted_by_ref = component_ref_get_counted_by (v);
>> +
>> while (v != pt_var && TREE_CODE (v) == COMPONENT_REF)
>> if (TREE_CODE (TREE_TYPE (TREE_OPERAND (v, 0)))
>> != UNION_TYPE
>> @@ -673,8 +678,11 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> == RECORD_TYPE)
>> {
>> /* compute object size only if v is not a
>> - flexible array member. */
>> - if (!is_flexible_array_mem_ref)
>> + flexible array member or the flexible array member
>> + has a known element count indicated by the user
>> + through attribute counted_by. */
>> + if (!is_flexible_array_mem_ref
>> + || counted_by_ref)
>> {
>> v = NULL_TREE;
>> break;
>> @@ -707,9 +715,24 @@ addr_object_size (struct object_size_info *osi, const_tree ptr,
>> if (var != pt_var)
>> {
>> - var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
>> - if (!TREE_CONSTANT (var_size))
>> - var_size = get_or_create_ssa_default_def (cfun, var_size);
>> + if (!counted_by_ref)
>> + {
>> + var_size = TYPE_SIZE_UNIT (TREE_TYPE (var));
>> + if (!TREE_CONSTANT (var_size))
>> + var_size = get_or_create_ssa_default_def (cfun, var_size);
>> + }
>> + else
>> + {
>> + gcc_assert (TREE_CODE (var) == COMPONENT_REF
>> + && TREE_CODE (TREE_TYPE (var)) == ARRAY_TYPE);
>> + tree element_size = TYPE_SIZE_UNIT (TREE_TYPE (TREE_TYPE (var)));
>> + var_size
>> + = size_binop (MULT_EXPR,
>> + fold_convert (sizetype, counted_by_ref),
>> + fold_convert (sizetype, element_size));
>> + if (!todo)
>> + todo = TODO_update_ssa_only_virtuals;
>> + }
>
> I feel like this could make a good separate function (get_subobject_size or something like that) to make it easier to read.
Will try this in the next version.
>
>> if (!var_size)
>> return false;
>> }
>> diff --git a/gcc/tree.cc b/gcc/tree.cc
>> index fcd36ae0cd74..3b6ddcbdcbf8 100644
>> --- a/gcc/tree.cc
>> +++ b/gcc/tree.cc
>> @@ -12745,6 +12745,32 @@ array_ref_element_size (tree exp)
>> return SUBSTITUTE_PLACEHOLDER_IN_EXPR (TYPE_SIZE_UNIT (elmt_type), exp);
>> }
>> +/* For a component_ref that has an array type ARRAY_REF, return TRUE when
>> + an counted_by attribute attached to the corresponding FIELD_DECL.
>> + return FALSE otherwise. */
>> +bool
>> +component_ref_has_counted_by_p (tree array_ref)
>> +{
>> + if (TREE_CODE (array_ref) != COMPONENT_REF)
>> + return false;
>> +
>> + if (TREE_CODE (TREE_TYPE (array_ref)) != ARRAY_TYPE)
>> + return false;
>> +
>> + tree struct_object = TREE_OPERAND (array_ref, 0);
>> + tree struct_type = TREE_TYPE (struct_object);
>> +
>> + if (!RECORD_OR_UNION_TYPE_P (struct_type))
>> + return false;
>> + tree field_decl = TREE_OPERAND (array_ref, 1);
>> + tree attr_counted_by = lookup_attribute ("counted_by",
>> + DECL_ATTRIBUTES (field_decl));
>> +
>> + if (!attr_counted_by)
>> + return false;
>> + return true;
>> +}
>> +
>> /* Given a field list, FIELDLIST, of a structure/union, return a TREE_LIST,
>> with each TREE_VALUE a FIELD_DECL stepping down the chain to the FIELD
>> whose name is FIELDNAME, which is the last TREE_VALUE of the list.
>> @@ -12771,7 +12797,7 @@ get_named_field (tree fieldlist, const char *fieldname)
>> fields inside it recursively. */
>> else if (RECORD_OR_UNION_TYPE_P (TREE_TYPE (field)))
>> if ((named_field = get_named_field (TYPE_FIELDS (TREE_TYPE (field)),
>> - fieldname)) != NULL_TREE)
>> + fieldname)) != NULL_TREE)
>
> Unrelated whitespace change?
Will check on this and fix it in the next version.
>
>> {
>> named_field = tree_cons (NULL_TREE, field, named_field);
>> break;
>> @@ -12784,6 +12810,73 @@ get_named_field (tree fieldlist, const char *fieldname)
>> return named_field;
>> }
>> +/* For a component_ref that has an array type ARRAY_REF, get the object that
>> + represents its counted_by per the attribute counted_by attached to
>> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
>> + object.
>> + For example, if:
>> +
>> + struct P {
>> + int k;
>> + int x[] __attribute__ ((counted_by (k)));
>> + } *p;
>> +
>> + for the following reference:
>> +
>> + p->x[b]
>> +
>> + the object that represents its element count will be:
>> +
>> + p->k
>> +
>> + So, when component_ref_get_counted_by (p->x[b]) is called, p->k should be
>> + returned.
>> +*/
>> +
>> +tree
>> +component_ref_get_counted_by (tree array_ref)
>> +{
>> + if (! component_ref_has_counted_by_p (array_ref))
>> + return NULL_TREE;
>> +
>> + tree struct_object = TREE_OPERAND (array_ref, 0);
>> + tree struct_type = TREE_TYPE (struct_object);
>> + tree field_decl = TREE_OPERAND (array_ref, 1);
>> + tree attr_counted_by = lookup_attribute ("counted_by",
>> + DECL_ATTRIBUTES (field_decl));
>> + gcc_assert (attr_counted_by);
>> +
>> + /* If there is an counted_by attribute attached to the field,
>> + get the field that maps to the counted_by. */
>> +
>> + const char *fieldname
>> + = IDENTIFIER_POINTER (TREE_VALUE (TREE_VALUE (attr_counted_by)));
>> +
>> + tree counted_by_field = get_named_field (TYPE_FIELDS (struct_type),
>> + fieldname);
>> +
>> + gcc_assert (counted_by_field);
>> +
>> + /* generate the tree node that represent the counted_by of this array
>
> Capitalize first word. Also s/represent/represents/
Okay.
>
>> + ref. This is a (possible nested) COMPONENT_REF to the counted_by_field
>
> possibly nested
Okay.
>
>> + of the containing structure. */
>> +
>> + tree counted_by_ref = NULL_TREE;
>> + tree object = struct_object;
>> + do
>> + {
>> + tree field = TREE_VALUE (counted_by_field);
>> +
>> + counted_by_ref = build3 (COMPONENT_REF,
>> + TREE_TYPE (field),
>> + unshare_expr (object), field,
>> + NULL_TREE);
>> + object = counted_by_ref;
>> + counted_by_field = TREE_CHAIN (counted_by_field);
>> + }
>> + while (counted_by_field);
>> + return counted_by_ref;
>> +}
>> /* Return a tree representing the lower bound of the array mentioned in
>> EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> diff --git a/gcc/tree.h b/gcc/tree.h
>> index 4859becaa1e7..07eed7219835 100644
>> --- a/gcc/tree.h
>> +++ b/gcc/tree.h
>> @@ -5619,11 +5619,21 @@ extern tree get_base_address (tree t);
>> of EXP, an ARRAY_REF or an ARRAY_RANGE_REF. */
>> extern tree array_ref_element_size (tree);
>> +/* Give a component_ref that has an array type, return true when an
>> + attribute counted_by attached to the corresponding FIELD_DECL. */
>> +extern bool component_ref_has_counted_by_p (tree);
>> +
>> /* Given a field list, FIELDLIST, of a structure/union, return the FIELD whose
>> name is FIELDNAME, return NULL_TREE if such field is not found.
>> searching nested anonymous structure/union recursively. */
>> extern tree get_named_field (tree, const char *);
>> +/* Give a component_ref that has an array type, return the object that
>> + represents its counted_by per the attribute counted_by attached to
>> + the corresponding FIELD_DECL. return NULL_TREE when cannot find such
>> + object. */
>> +extern tree component_ref_get_counted_by (tree);
>> +
>> /* Return a typenode for the "standard" C type with a given name. */
>> extern tree get_typenode_from_name (const char *);
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-05 20:08 ` Siddhesh Poyarekar
2023-10-05 22:35 ` Kees Cook
@ 2023-10-18 21:11 ` Qing Zhao
2023-10-19 23:33 ` Kees Cook
2023-10-20 17:08 ` HELP: Will the reordering happen? " Qing Zhao
1 sibling, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-18 21:11 UTC (permalink / raw)
To: Siddhesh Poyarekar, richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
> On Oct 5, 2023, at 4:08 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-08-25 11:24, Qing Zhao wrote:
>> This is the 3rd version of the patch, per our discussion based on the
>> review comments for the 1st and 2nd version, the major changes in this
>> version are:
>
> Hi Qing,
>
> I hope the review was helpful. Overall, a couple of things to consider:
>
> 1. How would you handle potential reordering between assignment of the size to the counted_by field with the __bdos call that may consume it? You'll probably need to express some kind of dependency there or in the worst case, insert a barrier to disallow reordering.
Good point!
So, your example in the respond to [V3][PATCH 2/3]Use the counted_by atribute info in builtin object size [PR108896]:
“
Maybe another test where the allocation, size assignment and __bdos call happen in the same function, where the allocator is not recognized by gcc:
void *
__attribute__ ((noinline))
alloc (size_t sz)
{
return __builtin_malloc (sz);
}
void test (size_t sz)
{
array_annotated = alloc (sz);
array_annotated->b = sz;
return __builtin_dynamic_object_size (array_annotated->c, 1);
}
The interesting thing to test (and ensure in the codegen) is that the assignment to array_annotated->b does not get reordered to below the __builtin_dynamic_object_size call since technically there is no data dependency between the two.
“
Will test on this.
Not sure whether the current GCC alias analysis is able to distinguish one field of a structure from another field of the same structure, if YES, then
We need to add an explicit dependency edge from the write to “array_annotated->b” to the call to “__builtin_dynamic_object_size(array_annotated->c,1)”.
I will check on this and see how to resolve this issue.
I guess the possible solution is that we can add an implicit ref to “array_annotated->b” at the call to “__builtin_dynamic_object_size(array_annotated->c, 1)” if the counted_by attribute is available. That should resolve the issue.
Richard, what do you think on this?
>
> 2. How would you handle signedness of the size field? The size gets converted to sizetype everywhere it is used and overflows/underflows may produce interesting results. Do you want to limit the types to unsigned or do you want to add a disclaimer in the docs? The former seems like the *right* thing to do given that it is a new feature; best to enforce the cleaner habit at the outset.
As I replied to Martin in another email, I plan to do the following to resolve this issue:
1. No specification for signed or unsigned for counted_by field.
2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
Then, we will be consistent with the handling of VLA.
So, I will not change anything for the current patch.
However, I will add the sanitizer option in a followup patch set.
Let me know your opinion.
thanks.
Qing
>
> Thanks,
> Sid
>
>> ***Against 1st version:
>> 1. change the name "element_count" to "counted_by";
>> 2. change the parameter for the attribute from a STRING to an
>> Identifier;
>> 3. Add logic and testing cases to handle anonymous structure/unions;
>> 4. Clarify documentation to permit the situation when the allocation
>> size is larger than what's specified by "counted_by", at the same time,
>> it's user's error if allocation size is smaller than what's specified by
>> "counted_by";
>> 5. Add a complete testing case for using counted_by attribute in
>> __builtin_dynamic_object_size when there is mismatch between the
>> allocation size and the value of "counted_by", the expecting behavior
>> for each case and the explanation on why in the comments.
>> ***Against 2rd version:
>> 1. Identify a tree node sharing issue and fixed it in the routine
>> "component_ref_get_counted_ty" of tree.cc;
>> 2. Update the documentation and testing cases with the clear usage
>> of the fomula to compute the allocation size:
>> MAX (sizeof (struct A), offsetof (struct A, array[0]) + counted_by * sizeof(element))
>> (the algorithm used in tree-object-size.cc is correct).
>> In this set of patches, the major functionality provided is:
>> 1. a new attribute "counted_by";
>> 2. use this new attribute in bound sanitizer;
>> 3. use this new attribute in dynamic object size for subobject size;
>> As discussed, I plan to add two more separate patches sets after this initial
>> patch set is approved and committed.
>> set 1. A new warning option and a new sanitizer option for the user error
>> when the allocation size is smaller than the value of "counted_by".
>> set 2. An improvement to __builtin_dynamic_object_size for whole-object
>> size of the structure with FAM annaoted with counted_by.
>> there are also some existing bugs in tree-object-size.cc identified
>> during the study, and PRs were filed to record them. these bugs will
>> be fixed seperately with individual patches:
>> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111030
>> https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111040
>> Bootstrapped and regression tested on both aarch64 and X86, no issue.
>> Please see more details on the description of this work on:
>> https://gcc.gnu.org/pipermail/gcc-patches/2023-May/619708.html
>> and more discussions on
>> https://gcc.gnu.org/pipermail/gcc-patches/2023-August/626376.html
>> Okay for committing?
>> thanks.
>> Qing
>> Qing Zhao (3):
>> Provide counted_by attribute to flexible array member field (PR108896)
>> Use the counted_by atribute info in builtin object size [PR108896]
>> Use the counted_by attribute information in bound sanitizer[PR108896]
>> gcc/c-family/c-attribs.cc | 54 ++++-
>> gcc/c-family/c-common.cc | 13 ++
>> gcc/c-family/c-common.h | 1 +
>> gcc/c-family/c-ubsan.cc | 16 ++
>> gcc/c/c-decl.cc | 79 +++++--
>> gcc/doc/extend.texi | 77 +++++++
>> .../gcc.dg/flex-array-counted-by-2.c | 74 ++++++
>> .../gcc.dg/flex-array-counted-by-3.c | 210 ++++++++++++++++++
>> gcc/testsuite/gcc.dg/flex-array-counted-by.c | 40 ++++
>> .../ubsan/flex-array-counted-by-bounds-2.c | 27 +++
>> .../ubsan/flex-array-counted-by-bounds.c | 46 ++++
>> gcc/tree-object-size.cc | 37 ++-
>> gcc/tree.cc | 133 +++++++++++
>> gcc/tree.h | 15 ++
>> 14 files changed, 797 insertions(+), 25 deletions(-)
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-2.c
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by-3.c
>> create mode 100644 gcc/testsuite/gcc.dg/flex-array-counted-by.c
>> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds-2.c
>> create mode 100644 gcc/testsuite/gcc.dg/ubsan/flex-array-counted-by-bounds.c
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-18 21:11 ` Qing Zhao
@ 2023-10-19 23:33 ` Kees Cook
2023-10-20 9:50 ` Martin Uecker
2023-10-20 17:08 ` HELP: Will the reordering happen? " Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-19 23:33 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, Martin Uecker, isanbard
On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> As I replied to Martin in another email, I plan to do the following to resolve this issue:
>
> 1. No specification for signed or unsigned for counted_by field.
> 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
I don't understand why this needs to be a runtime sanitizer. The
signedness is known at compile time, so I would expect a -W option. Or
do you mean you'd split up -fsanitize=bounds between unsigned and signed
indexes? I'd find that kind of awkward for the kernel... but I feel like
I've misunderstood something. :)
-Kees
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-19 23:33 ` Kees Cook
@ 2023-10-20 9:50 ` Martin Uecker
2023-10-20 18:34 ` Kees Cook
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-20 9:50 UTC (permalink / raw)
To: Kees Cook, Qing Zhao
Cc: Siddhesh Poyarekar, richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, isanbard
Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> >
> > 1. No specification for signed or unsigned for counted_by field.
> > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
>
> I don't understand why this needs to be a runtime sanitizer. The
> signedness is known at compile time, so I would expect a -W option.
The signedness of the type but not of the value.
But I would not want to have a warning for signed
counter types by default because I would prefer
to use signed types (for various reasons including
better overflow detection).
> Or
> do you mean you'd split up -fsanitize=bounds between unsigned and signed
> indexes? I'd find that kind of awkward for the kernel... but I feel like
> I've misunderstood something. :)
>
> -Kees
The idea would be to detect at run-time the case
if x->buf is used at a time where x->counter
is negative and also when x->counter * sizeof(x->buf[0])
overflows or is too big.
This would be similar to
int a[n];
where it is detected at run-time if n is not-positive.
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-18 21:11 ` Qing Zhao
2023-10-19 23:33 ` Kees Cook
@ 2023-10-20 17:08 ` Qing Zhao
2023-10-20 18:22 ` Richard Biener
1 sibling, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-20 17:08 UTC (permalink / raw)
To: Siddhesh Poyarekar, richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
Sid,
(Richard, can you please help me to make sure this? Thanks a lot)
I studied a little bit more on the following question you raised during the review process:
For the following small testing case:
1 struct annotated {
2 int foo;
3 char array[] __attribute__((counted_by (foo)));
4 };
5
6 extern struct annotated * alloc_buf (int);
7
8 int test (int sz)
9 {
10 struct annotated * array_annotated = alloc_buf (sz);
11 array_annotated->foo = sz;
12 return __builtin_dynamic_object_size (array_annotated->array, 1);
13 }
Whether the assignment of the size to the counted_by field at line 11 and the consumer of the size at line 12 at call to __bdos might be reordered by GCC?
The following is my thought:
1. _bdos computation passes (both pass_early_object_sizes and pass_object_sizes) are in the early stage of SSA optimizations. In which, pass_early_object_sizes happens before almost all the optimizations, no reordering is possible in this pass;
2. Then how about the pass “pass_object_sizes”?
Immediately after the pass_build_ssa, the IR for the routine “test” is with the SSA form: (compiled with -O3):
1 int test (int sz)
2 {
3 struct annotated * array_annotated;
4 char[0:] * _1;
5 long unsigned int _2;
6 int _8;
7
8 <bb 2> :
9 array_annotated_6 = alloc_buf (sz_4(D));
10 array_annotated_6->foo = sz_4(D);
11 _1 = &array_annotated_6->array;
12 _2 = __builtin_dynamic_object_size (_1, 1);
13 _8 = (int) _2;
14 return _8;
15 }
In the above IR, the key portion is line 10 and line 11: (whether these two lines might be reordered with SSA optimization?)
10 array_annotated_6->foo = sz_4(D);
11 _1 = &array_annotated_6->array;
The major question here is: whether the SSA optimizations are able to distinguish the object “array_annotated_6->foo” at line 10 is independent with
the object “array_annotated-_6->array” at line 11?
If the SSA optimizations can distinguish “array_annotated_6->foo” from “array_annotated_6->array”, then these two lines might be reordered.
Otherwise, these two lines will not be reordered by SSA optimizations.
I am not very familiar with the details of the SSA optimizations, but my guess is, two fields of the same structure might not be distinguished by the SSA optimizations, then line 10 and line 11 will not be reordered by SSA optimizations.
Richard, is my guess correct?
Thanks a lot for your help.
Qing
>> On Oct 5, 2023, at 4:08 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>
>> I hope the review was helpful. Overall, a couple of things to consider:
>>
>> 1. How would you handle potential reordering between assignment of the size to the counted_by field with the __bdos call that may consume it? You'll probably need to express some kind of dependency there or in the worst case, insert a barrier to disallow reordering.
>
> Good point!
>
> So, your example in the respond to [V3][PATCH 2/3]Use the counted_by atribute info in builtin object size [PR108896]:
> “
> Maybe another test where the allocation, size assignment and __bdos call happen in the same function, where the allocator is not recognized by gcc:
>
> void *
> __attribute__ ((noinline))
> alloc (size_t sz)
> {
> return __builtin_malloc (sz);
> }
>
> void test (size_t sz)
> {
> array_annotated = alloc (sz);
> array_annotated->b = sz;
> return __builtin_dynamic_object_size (array_annotated->c, 1);
> }
>
> The interesting thing to test (and ensure in the codegen) is that the assignment to array_annotated->b does not get reordered to below the __builtin_dynamic_object_size call since technically there is no data dependency between the two.
> “
> Will test on this.
>
> Not sure whether the current GCC alias analysis is able to distinguish one field of a structure from another field of the same structure, if YES, then
> We need to add an explicit dependency edge from the write to “array_annotated->b” to the call to “__builtin_dynamic_object_size(array_annotated->c,1)”.
> I will check on this and see how to resolve this issue.
>
> I guess the possible solution is that we can add an implicit ref to “array_annotated->b” at the call to “__builtin_dynamic_object_size(array_annotated->c, 1)” if the counted_by attribute is available. That should resolve the issue.
>
> Richard, what do you think on this?
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 17:08 ` HELP: Will the reordering happen? " Qing Zhao
@ 2023-10-20 18:22 ` Richard Biener
2023-10-20 18:38 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-20 18:22 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> Am 20.10.2023 um 19:09 schrieb Qing Zhao <qing.zhao@oracle.com>:
>
> Sid,
>
> (Richard, can you please help me to make sure this? Thanks a lot)
>
> I studied a little bit more on the following question you raised during the review process:
>
> For the following small testing case:
>
> 1 struct annotated {
> 2 int foo;
> 3 char array[] __attribute__((counted_by (foo)));
> 4 };
> 5
> 6 extern struct annotated * alloc_buf (int);
> 7
> 8 int test (int sz)
> 9 {
> 10 struct annotated * array_annotated = alloc_buf (sz);
> 11 array_annotated->foo = sz;
> 12 return __builtin_dynamic_object_size (array_annotated->array, 1);
> 13 }
>
> Whether the assignment of the size to the counted_by field at line 11 and the consumer of the size at line 12 at call to __bdos might be reordered by GCC?
>
> The following is my thought:
>
> 1. _bdos computation passes (both pass_early_object_sizes and pass_object_sizes) are in the early stage of SSA optimizations. In which, pass_early_object_sizes happens before almost all the optimizations, no reordering is possible in this pass;
>
> 2. Then how about the pass “pass_object_sizes”?
>
> Immediately after the pass_build_ssa, the IR for the routine “test” is with the SSA form: (compiled with -O3):
>
> 1 int test (int sz)
> 2 {
> 3 struct annotated * array_annotated;
> 4 char[0:] * _1;
> 5 long unsigned int _2;
> 6 int _8;
> 7
> 8 <bb 2> :
> 9 array_annotated_6 = alloc_buf (sz_4(D));
> 10 array_annotated_6->foo = sz_4(D);
> 11 _1 = &array_annotated_6->array;
> 12 _2 = __builtin_dynamic_object_size (_1, 1);
> 13 _8 = (int) _2;
> 14 return _8;
> 15 }
>
> In the above IR, the key portion is line 10 and line 11: (whether these two lines might be reordered with SSA optimization?)
>
> 10 array_annotated_6->foo = sz_4(D);
> 11 _1 = &array_annotated_6->array;
>
> The major question here is: whether the SSA optimizations are able to distinguish the object “array_annotated_6->foo” at line 10 is independent with
> the object “array_annotated-_6->array” at line 11?
>
> If the SSA optimizations can distinguish “array_annotated_6->foo” from “array_annotated_6->array”, then these two lines might be reordered.
> Otherwise, these two lines will not be reordered by SSA optimizations.
>
> I am not very familiar with the details of the SSA optimizations, but my guess is, two fields of the same structure might not be distinguished by the SSA optimizations, then line 10 and line 11 will not be reordered by SSA optimizations.
>
> Richard, is my guess correct?
There is no data dependence between the memory access and the address computation so nothing prevents the reordering. If you put another same bos call before the access I expect the addresses to be CSEd, effectively moving the later before the access.
Richard
> Thanks a lot for your help.
>
> Qing
>
>>>> On Oct 5, 2023, at 4:08 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>
>>> I hope the review was helpful. Overall, a couple of things to consider:
>>>
>>> 1. How would you handle potential reordering between assignment of the size to the counted_by field with the __bdos call that may consume it? You'll probably need to express some kind of dependency there or in the worst case, insert a barrier to disallow reordering.
>>
>> Good point!
>>
>> So, your example in the respond to [V3][PATCH 2/3]Use the counted_by atribute info in builtin object size [PR108896]:
>> “
>> Maybe another test where the allocation, size assignment and __bdos call happen in the same function, where the allocator is not recognized by gcc:
>>
>> void *
>> __attribute__ ((noinline))
>> alloc (size_t sz)
>> {
>> return __builtin_malloc (sz);
>> }
>>
>> void test (size_t sz)
>> {
>> array_annotated = alloc (sz);
>> array_annotated->b = sz;
>> return __builtin_dynamic_object_size (array_annotated->c, 1);
>> }
>>
>> The interesting thing to test (and ensure in the codegen) is that the assignment to array_annotated->b does not get reordered to below the __builtin_dynamic_object_size call since technically there is no data dependency between the two.
>> “
>> Will test on this.
>>
>> Not sure whether the current GCC alias analysis is able to distinguish one field of a structure from another field of the same structure, if YES, then
>> We need to add an explicit dependency edge from the write to “array_annotated->b” to the call to “__builtin_dynamic_object_size(array_annotated->c,1)”.
>> I will check on this and see how to resolve this issue.
>>
>> I guess the possible solution is that we can add an implicit ref to “array_annotated->b” at the call to “__builtin_dynamic_object_size(array_annotated->c, 1)” if the counted_by attribute is available. That should resolve the issue.
>>
>> Richard, what do you think on this?
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 9:50 ` Martin Uecker
@ 2023-10-20 18:34 ` Kees Cook
2023-10-20 18:48 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-20 18:34 UTC (permalink / raw)
To: Martin Uecker
Cc: Qing Zhao, Siddhesh Poyarekar, richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
> Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> > On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> > >
> > > 1. No specification for signed or unsigned for counted_by field.
> > > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
> >
> > I don't understand why this needs to be a runtime sanitizer. The
> > signedness is known at compile time, so I would expect a -W option.
>
> The signedness of the type but not of the value.
>
> But I would not want to have a warning for signed
> counter types by default because I would prefer
> to use signed types (for various reasons including
> better overflow detection).
>
> > Or
> > do you mean you'd split up -fsanitize=bounds between unsigned and signed
> > indexes? I'd find that kind of awkward for the kernel... but I feel like
> > I've misunderstood something. :)
> >
> > -Kees
>
> The idea would be to detect at run-time the case
> if x->buf is used at a time where x->counter
> is negative and also when x->counter * sizeof(x->buf[0])
> overflows or is too big.
>
> This would be similar to
>
> int a[n];
>
> where it is detected at run-time if n is not-positive.
Right. I guess what I mean to say is that I would expect this case to
already be caught by -fsanitize=bounds -- I don't see a reason to add an
additional sanitizer option.
struct foo {
int count;
int array[] __counted_by(count);
};
foo->count = 5;
foo->array[0] = 1; // ok
foo->array[10] = 1; // -fsanitize=bounds will catch this
foo->array[-10] = 1; // -fsanitize=bounds will catch this too
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 18:22 ` Richard Biener
@ 2023-10-20 18:38 ` Qing Zhao
2023-10-20 19:10 ` Siddhesh Poyarekar
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-20 18:38 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> On Oct 20, 2023, at 2:22 PM, Richard Biener <richard.guenther@gmail.com> wrote:
>
>
>
>> Am 20.10.2023 um 19:09 schrieb Qing Zhao <qing.zhao@oracle.com>:
>>
>> Sid,
>>
>> (Richard, can you please help me to make sure this? Thanks a lot)
>>
>> I studied a little bit more on the following question you raised during the review process:
>>
>> For the following small testing case:
>>
>> 1 struct annotated {
>> 2 int foo;
>> 3 char array[] __attribute__((counted_by (foo)));
>> 4 };
>> 5
>> 6 extern struct annotated * alloc_buf (int);
>> 7
>> 8 int test (int sz)
>> 9 {
>> 10 struct annotated * array_annotated = alloc_buf (sz);
>> 11 array_annotated->foo = sz;
>> 12 return __builtin_dynamic_object_size (array_annotated->array, 1);
>> 13 }
>>
>> Whether the assignment of the size to the counted_by field at line 11 and the consumer of the size at line 12 at call to __bdos might be reordered by GCC?
>>
>> The following is my thought:
>>
>> 1. _bdos computation passes (both pass_early_object_sizes and pass_object_sizes) are in the early stage of SSA optimizations. In which, pass_early_object_sizes happens before almost all the optimizations, no reordering is possible in this pass;
>>
>> 2. Then how about the pass “pass_object_sizes”?
>>
>> Immediately after the pass_build_ssa, the IR for the routine “test” is with the SSA form: (compiled with -O3):
>>
>> 1 int test (int sz)
>> 2 {
>> 3 struct annotated * array_annotated;
>> 4 char[0:] * _1;
>> 5 long unsigned int _2;
>> 6 int _8;
>> 7
>> 8 <bb 2> :
>> 9 array_annotated_6 = alloc_buf (sz_4(D));
>> 10 array_annotated_6->foo = sz_4(D);
>> 11 _1 = &array_annotated_6->array;
>> 12 _2 = __builtin_dynamic_object_size (_1, 1);
>> 13 _8 = (int) _2;
>> 14 return _8;
>> 15 }
>>
>> In the above IR, the key portion is line 10 and line 11: (whether these two lines might be reordered with SSA optimization?)
>>
>> 10 array_annotated_6->foo = sz_4(D);
>> 11 _1 = &array_annotated_6->array;
>>
>> The major question here is: whether the SSA optimizations are able to distinguish the object “array_annotated_6->foo” at line 10 is independent with
>> the object “array_annotated-_6->array” at line 11?
>>
>> If the SSA optimizations can distinguish “array_annotated_6->foo” from “array_annotated_6->array”, then these two lines might be reordered.
>> Otherwise, these two lines will not be reordered by SSA optimizations.
>>
>> I am not very familiar with the details of the SSA optimizations, but my guess is, two fields of the same structure might not be distinguished by the SSA optimizations, then line 10 and line 11 will not be reordered by SSA optimizations.
>>
>> Richard, is my guess correct?
>
> There is no data dependence between the memory access and the address computation so nothing prevents the reordering.
Okay, I see. then:
10 array_annotated_6->foo = sz_4(D);
11 _1 = &array_annotated_6->array;
Line 10 and line 11 could be reordered.
And then
10 array_annotated_6->foo = sz_4(D);
12 _2 = __builtin_dynamic_object_size (_1, 1);
Line 10 and 12 could be reordered too.
Then what’s the best way to add such data dependence in the IR?
How about the following:
Add one more parameter to __builtin_dynamic_object_size(), i.e
__builtin_dynamic_object_size (_1,1,array_annotated->foo)?
When we see the structure field has counted_by attribute.
Then we can enforce such data dependence and avoid potential reordering.
What’s your opinion? Do you have other suggestion on the solution?
Qing
If you put another same bos call before the access I expect the addresses to be CSEd, effectively moving the later before the access.
>
> Richard
>
>> Thanks a lot for your help.
>>
>> Qing
>>
>>>>> On Oct 5, 2023, at 4:08 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>
>>>> I hope the review was helpful. Overall, a couple of things to consider:
>>>>
>>>> 1. How would you handle potential reordering between assignment of the size to the counted_by field with the __bdos call that may consume it? You'll probably need to express some kind of dependency there or in the worst case, insert a barrier to disallow reordering.
>>>
>>> Good point!
>>>
>>> So, your example in the respond to [V3][PATCH 2/3]Use the counted_by atribute info in builtin object size [PR108896]:
>>> “
>>> Maybe another test where the allocation, size assignment and __bdos call happen in the same function, where the allocator is not recognized by gcc:
>>>
>>> void *
>>> __attribute__ ((noinline))
>>> alloc (size_t sz)
>>> {
>>> return __builtin_malloc (sz);
>>> }
>>>
>>> void test (size_t sz)
>>> {
>>> array_annotated = alloc (sz);
>>> array_annotated->b = sz;
>>> return __builtin_dynamic_object_size (array_annotated->c, 1);
>>> }
>>>
>>> The interesting thing to test (and ensure in the codegen) is that the assignment to array_annotated->b does not get reordered to below the __builtin_dynamic_object_size call since technically there is no data dependency between the two.
>>> “
>>> Will test on this.
>>>
>>> Not sure whether the current GCC alias analysis is able to distinguish one field of a structure from another field of the same structure, if YES, then
>>> We need to add an explicit dependency edge from the write to “array_annotated->b” to the call to “__builtin_dynamic_object_size(array_annotated->c,1)”.
>>> I will check on this and see how to resolve this issue.
>>>
>>> I guess the possible solution is that we can add an implicit ref to “array_annotated->b” at the call to “__builtin_dynamic_object_size(array_annotated->c, 1)” if the counted_by attribute is available. That should resolve the issue.
>>>
>>> Richard, what do you think on this?
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 18:34 ` Kees Cook
@ 2023-10-20 18:48 ` Qing Zhao
2023-10-20 19:54 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-20 18:48 UTC (permalink / raw)
To: Kees Cook, Martin Uecker
Cc: Siddhesh Poyarekar, richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, isanbard
> On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
>
> On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
>> Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
>>> On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
>>>> As I replied to Martin in another email, I plan to do the following to resolve this issue:
>>>>
>>>> 1. No specification for signed or unsigned for counted_by field.
>>>> 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
>>>
>>> I don't understand why this needs to be a runtime sanitizer. The
>>> signedness is known at compile time, so I would expect a -W option.
>>
>> The signedness of the type but not of the value.
>>
>> But I would not want to have a warning for signed
>> counter types by default because I would prefer
>> to use signed types (for various reasons including
>> better overflow detection).
>>
>>> Or
>>> do you mean you'd split up -fsanitize=bounds between unsigned and signed
>>> indexes? I'd find that kind of awkward for the kernel... but I feel like
>>> I've misunderstood something. :)
>>>
>>> -Kees
>>
>> The idea would be to detect at run-time the case
>> if x->buf is used at a time where x->counter
>> is negative and also when x->counter * sizeof(x->buf[0])
>> overflows or is too big.
>>
>> This would be similar to
>>
>> int a[n];
>>
>> where it is detected at run-time if n is not-positive.
>
> Right. I guess what I mean to say is that I would expect this case to
> already be caught by -fsanitize=bounds -- I don't see a reason to add an
> additional sanitizer option.
>
> struct foo {
> int count;
> int array[] __counted_by(count);
> };
>
> foo->count = 5;
> foo->array[0] = 1; // ok
> foo->array[10] = 1; // -fsanitize=bounds will catch this
> foo->array[-10] = 1; // -fsanitize=bounds will catch this too
>
>
just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
Qing
> --
> Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 18:38 ` Qing Zhao
@ 2023-10-20 19:10 ` Siddhesh Poyarekar
2023-10-20 20:41 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-20 19:10 UTC (permalink / raw)
To: Qing Zhao, Richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
On 2023-10-20 14:38, Qing Zhao wrote:
> How about the following:
>
> Add one more parameter to __builtin_dynamic_object_size(), i.e
>
> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>
> When we see the structure field has counted_by attribute.
Or maybe add a barrier preventing any assignments to
array_annotated->foo from being reordered below the __bdos call?
Basically an __asm__ with array_annotated->foo in the clobber list ought
to do it I think.
It may not work for something like this though:
static size_t
get_size_of (void *ptr)
{
return __bdos (ptr, 1);
}
void
foo (size_t sz)
{
array_annotated = __builtin_malloc (sz);
array_annotated = sz;
...
__builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
...
}
because the call to get_size_of () may not have been inlined that early.
The more fool-proof alternative may be to put a compile time barrier
right below the assignment to array_annotated->foo; I reckon you could
do that early in the front end by marking the size identifier and then
tracking assignments to that identifier. That may have a slight runtime
performance overhead since it may prevent even legitimate reordering. I
can't think of another alternative at the moment...
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 18:48 ` Qing Zhao
@ 2023-10-20 19:54 ` Martin Uecker
2023-10-23 18:17 ` Qing Zhao
2023-10-23 19:52 ` Kees Cook
0 siblings, 2 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-20 19:54 UTC (permalink / raw)
To: Qing Zhao, Kees Cook; +Cc: gcc-patches
Am Freitag, dem 20.10.2023 um 18:48 +0000 schrieb Qing Zhao:
>
> > On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
> >
> > On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
> > > Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> > > > On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > > > > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> > > > >
> > > > > 1. No specification for signed or unsigned for counted_by field.
> > > > > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
> > > >
> > > > I don't understand why this needs to be a runtime sanitizer. The
> > > > signedness is known at compile time, so I would expect a -W option.
> > >
> > > The signedness of the type but not of the value.
> > >
> > > But I would not want to have a warning for signed
> > > counter types by default because I would prefer
> > > to use signed types (for various reasons including
> > > better overflow detection).
> > >
> > > > Or
> > > > do you mean you'd split up -fsanitize=bounds between unsigned and signed
> > > > indexes? I'd find that kind of awkward for the kernel... but I feel like
> > > > I've misunderstood something. :)
> > > >
> > > > -Kees
> > >
> > > The idea would be to detect at run-time the case
> > > if x->buf is used at a time where x->counter
> > > is negative and also when x->counter * sizeof(x->buf[0])
> > > overflows or is too big.
> > >
> > > This would be similar to
> > >
> > > int a[n];
> > >
> > > where it is detected at run-time if n is not-positive.
> >
> > Right. I guess what I mean to say is that I would expect this case to
> > already be caught by -fsanitize=bounds -- I don't see a reason to add an
> > additional sanitizer option.
> >
> > struct foo {
> > int count;
> > int array[] __counted_by(count);
> > };
> >
> > foo->count = 5;
> > foo->array[0] = 1; // ok
> > foo->array[10] = 1; // -fsanitize=bounds will catch this
> > foo->array[-10] = 1; // -fsanitize=bounds will catch this too
> >
> >
>
> just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
>
> ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
> ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
>
Yes, but I thought we were discussing the case where count is
set to a negative value:
foo->count = -1;
int x = foo->array[3]; // UBSan should diagnose this
And also the case when foo->array becomes too big.
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 19:10 ` Siddhesh Poyarekar
@ 2023-10-20 20:41 ` Qing Zhao
2023-10-23 7:57 ` Richard Biener
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-20 20:41 UTC (permalink / raw)
To: Siddhesh Poyarekar, Richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-20 14:38, Qing Zhao wrote:
>> How about the following:
>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>> When we see the structure field has counted_by attribute.
>
> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
This might be the simplest solution?
Qing
>
> It may not work for something like this though:
>
> static size_t
> get_size_of (void *ptr)
> {
> return __bdos (ptr, 1);
> }
>
> void
> foo (size_t sz)
> {
> array_annotated = __builtin_malloc (sz);
> array_annotated = sz;
>
> ...
> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
> ...
> }
>
> because the call to get_size_of () may not have been inlined that early.
>
> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 20:41 ` Qing Zhao
@ 2023-10-23 7:57 ` Richard Biener
2023-10-23 11:27 ` Siddhesh Poyarekar
2023-10-23 14:56 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Richard Biener @ 2023-10-23 7:57 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>
>
>
> > On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
> >
> > On 2023-10-20 14:38, Qing Zhao wrote:
> >> How about the following:
> >> Add one more parameter to __builtin_dynamic_object_size(), i.e
> >> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
> >> When we see the structure field has counted_by attribute.
> >
> > Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>
> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>
> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>
> This might be the simplest solution?
If the dynamic object size is derived of a field then I think you need to
put the "load" of that memory location at the point (as argument)
of the __bos call right at parsing time. I know that's awkward because
you try to play tricks "discovering" that field only late, but that's not
going to work.
A related issue is that assignment to the field and storage allocation
are not tied together - if there's no use of the size data we might
remove the store of it as dead.
Of course I guess __bos then behaves like sizeof ().
Richard.
>
> Qing
>
> >
> > It may not work for something like this though:
> >
> > static size_t
> > get_size_of (void *ptr)
> > {
> > return __bdos (ptr, 1);
> > }
> >
> > void
> > foo (size_t sz)
> > {
> > array_annotated = __builtin_malloc (sz);
> > array_annotated = sz;
> >
> > ...
> > __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
> > ...
> > }
> >
> > because the call to get_size_of () may not have been inlined that early.
> >
> > The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
> >
> > Sid
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 7:57 ` Richard Biener
@ 2023-10-23 11:27 ` Siddhesh Poyarekar
2023-10-23 12:34 ` Richard Biener
2023-10-23 14:56 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-23 11:27 UTC (permalink / raw)
To: Richard Biener, Qing Zhao
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
On 2023-10-23 03:57, Richard Biener wrote:
> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>
>>
>>
>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>
>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>> How about the following:
>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>> When we see the structure field has counted_by attribute.
>>>
>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>
>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>
>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>
>> This might be the simplest solution?
>
> If the dynamic object size is derived of a field then I think you need to
> put the "load" of that memory location at the point (as argument)
> of the __bos call right at parsing time. I know that's awkward because
> you try to play tricks "discovering" that field only late, but that's not
> going to work.
>
> A related issue is that assignment to the field and storage allocation
> are not tied together - if there's no use of the size data we might
> remove the store of it as dead.
Maybe the trick then is to treat the size data as volatile? That ought
to discourage reordering and also prevent elimination of the "dead" store?
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 11:27 ` Siddhesh Poyarekar
@ 2023-10-23 12:34 ` Richard Biener
2023-10-23 13:23 ` Siddhesh Poyarekar
2023-10-23 15:14 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Richard Biener @ 2023-10-23 12:34 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Qing Zhao, Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
On Mon, Oct 23, 2023 at 1:27 PM Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-23 03:57, Richard Biener wrote:
> > On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
> >>
> >>
> >>
> >>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
> >>>
> >>> On 2023-10-20 14:38, Qing Zhao wrote:
> >>>> How about the following:
> >>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
> >>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
> >>>> When we see the structure field has counted_by attribute.
> >>>
> >>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
> >>
> >> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
> >>
> >> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
> >>
> >> This might be the simplest solution?
> >
> > If the dynamic object size is derived of a field then I think you need to
> > put the "load" of that memory location at the point (as argument)
> > of the __bos call right at parsing time. I know that's awkward because
> > you try to play tricks "discovering" that field only late, but that's not
> > going to work.
> >
> > A related issue is that assignment to the field and storage allocation
> > are not tied together - if there's no use of the size data we might
> > remove the store of it as dead.
>
> Maybe the trick then is to treat the size data as volatile? That ought
> to discourage reordering and also prevent elimination of the "dead" store?
But we are an optimizing compiler, not a static analysis machine, so I
fail to see how this is a useful suggestion.
I think Martins suggestion to approach this as a language extension
is more useful and would make it easier to handle this?
Richard.
> Thanks,
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 12:34 ` Richard Biener
@ 2023-10-23 13:23 ` Siddhesh Poyarekar
2023-10-23 15:14 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-23 13:23 UTC (permalink / raw)
To: Richard Biener
Cc: Qing Zhao, Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook,
Martin Uecker, isanbard
On 2023-10-23 08:34, Richard Biener wrote:
>>> A related issue is that assignment to the field and storage allocation
>>> are not tied together - if there's no use of the size data we might
>>> remove the store of it as dead.
>>
>> Maybe the trick then is to treat the size data as volatile? That ought
>> to discourage reordering and also prevent elimination of the "dead" store?
>
> But we are an optimizing compiler, not a static analysis machine, so I
> fail to see how this is a useful suggestion.
Sorry I didn't meant to suggest doing this in the middle-end.
> I think Martins suggestion to approach this as a language extension
> is more useful and would make it easier to handle this?
I think handling for this (e.g. treating any storage allocated for the
size member in the struct as volatile to prevent reordering or
elimination) would have to be implemented in the front-end, regardless
of whether it is a language extension or as a gcc attribute. How would
making it a language extension vs a gcc attribute make it different?
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 7:57 ` Richard Biener
2023-10-23 11:27 ` Siddhesh Poyarekar
@ 2023-10-23 14:56 ` Qing Zhao
2023-10-23 15:57 ` Richard Biener
1 sibling, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 14:56 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>
>>
>>
>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>
>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>> How about the following:
>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>> When we see the structure field has counted_by attribute.
>>>
>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>
>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>
>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>
>> This might be the simplest solution?
>
> If the dynamic object size is derived of a field then I think you need to
> put the "load" of that memory location at the point (as argument)
> of the __bos call right at parsing time. I know that's awkward because
> you try to play tricks "discovering" that field only late, but that's not
> going to work.
Is it better to do this at gimplification phase instead of FE?
VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
***During gimplification,
For a call to __builtin_dynamic_object_size (ptr, type)
Check whether the type of ptr includes counted_by attribute, if so, change the call to
__builtin_dynamic_object_size (ptr, type, counted_by field)
Then the correct data dependence should be represented well in the IR.
**During object size phase,
The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
>
> A related issue is that assignment to the field and storage allocation
> are not tied together
Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
> - if there's no use of the size data we might
> remove the store of it as dead.
Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
I guess that the compiler should be able to do this automatically?
thanks.
Qing
>
> Of course I guess __bos then behaves like sizeof ().
>
> Richard.
>
>>
>> Qing
>>
>>>
>>> It may not work for something like this though:
>>>
>>> static size_t
>>> get_size_of (void *ptr)
>>> {
>>> return __bdos (ptr, 1);
>>> }
>>>
>>> void
>>> foo (size_t sz)
>>> {
>>> array_annotated = __builtin_malloc (sz);
>>> array_annotated = sz;
>>>
>>> ...
>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>> ...
>>> }
>>>
>>> because the call to get_size_of () may not have been inlined that early.
>>>
>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>
>>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 12:34 ` Richard Biener
2023-10-23 13:23 ` Siddhesh Poyarekar
@ 2023-10-23 15:14 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 15:14 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> On Oct 23, 2023, at 8:34 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
> On Mon, Oct 23, 2023 at 1:27 PM Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>
>> On 2023-10-23 03:57, Richard Biener wrote:
>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>>
>>>>
>>>>
>>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>>
>>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>>> How about the following:
>>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>>> When we see the structure field has counted_by attribute.
>>>>>
>>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>>
>>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>>
>>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>>
>>>> This might be the simplest solution?
>>>
>>> If the dynamic object size is derived of a field then I think you need to
>>> put the "load" of that memory location at the point (as argument)
>>> of the __bos call right at parsing time. I know that's awkward because
>>> you try to play tricks "discovering" that field only late, but that's not
>>> going to work.
>>>
>>> A related issue is that assignment to the field and storage allocation
>>> are not tied together - if there's no use of the size data we might
>>> remove the store of it as dead.
>>
>> Maybe the trick then is to treat the size data as volatile? That ought
>> to discourage reordering and also prevent elimination of the "dead" store?
>
> But we are an optimizing compiler, not a static analysis machine, so I
> fail to see how this is a useful suggestion.
>
> I think Martins suggestion to approach this as a language extension
> is more useful and would make it easier to handle this?
I agree that making this as a language extension is a better and cleaner approach.
As we discussed before, the major issues with the language extension approach are:
1. Harder to be adopted by the existing source code due to the potential ABI/API change.
2. Much more effort and much longer time to be accepted.
In addition to the above issues, I guess the same issue exists even with a language extension,
Since for FMA, it’s the user (not the compiler) to allocate the storage for the FMA. (Should we
Also move this into compiler for the language extension? Then the existing source code need to
Be changed a lot to adopt the new language extension).
As a result, the size and the storage allocation cannot be guaranteed to be tied together too.
Qing
>
> Richard.
>
>> Thanks,
>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 14:56 ` Qing Zhao
@ 2023-10-23 15:57 ` Richard Biener
2023-10-23 16:37 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-23 15:57 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
>
>
>
>> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>
>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>
>>>
>>>
>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>
>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>> How about the following:
>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>> When we see the structure field has counted_by attribute.
>>>>
>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>
>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>
>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>
>>> This might be the simplest solution?
>>
>> If the dynamic object size is derived of a field then I think you need to
>> put the "load" of that memory location at the point (as argument)
>> of the __bos call right at parsing time. I know that's awkward because
>> you try to play tricks "discovering" that field only late, but that's not
>> going to work.
>
> Is it better to do this at gimplification phase instead of FE?
>
> VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
>
> For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
>
> ***During gimplification,
> For a call to __builtin_dynamic_object_size (ptr, type)
> Check whether the type of ptr includes counted_by attribute, if so, change the call to
> __builtin_dynamic_object_size (ptr, type, counted_by field)
>
> Then the correct data dependence should be represented well in the IR.
>
> **During object size phase,
>
> The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call. For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
Richard
>
>>
>> A related issue is that assignment to the field and storage allocation
>> are not tied together
>
> Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
>
> For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
>
>> - if there's no use of the size data we might
>> remove the store of it as dead.
>
> Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
> I guess that the compiler should be able to do this automatically?
>
> thanks.
>
> Qing
>>
>> Of course I guess __bos then behaves like sizeof ().
>>
>> Richard.
>>
>>>
>>> Qing
>>>
>>>>
>>>> It may not work for something like this though:
>>>>
>>>> static size_t
>>>> get_size_of (void *ptr)
>>>> {
>>>> return __bdos (ptr, 1);
>>>> }
>>>>
>>>> void
>>>> foo (size_t sz)
>>>> {
>>>> array_annotated = __builtin_malloc (sz);
>>>> array_annotated = sz;
>>>>
>>>> ...
>>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>>> ...
>>>> }
>>>>
>>>> because the call to get_size_of () may not have been inlined that early.
>>>>
>>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>>
>>>> Sid
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 15:57 ` Richard Biener
@ 2023-10-23 16:37 ` Qing Zhao
2023-10-23 18:06 ` Martin Uecker
2023-10-23 18:10 ` Joseph Myers
0 siblings, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 16:37 UTC (permalink / raw)
To: Richard Biener, Joseph Myers
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
> On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
>
>
>> Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
>>
>>
>>
>>> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>
>>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>>
>>>>
>>>>
>>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>>
>>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>>> How about the following:
>>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>>> When we see the structure field has counted_by attribute.
>>>>>
>>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>>
>>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>>
>>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>>
>>>> This might be the simplest solution?
>>>
>>> If the dynamic object size is derived of a field then I think you need to
>>> put the "load" of that memory location at the point (as argument)
>>> of the __bos call right at parsing time. I know that's awkward because
>>> you try to play tricks "discovering" that field only late, but that's not
>>> going to work.
>>
>> Is it better to do this at gimplification phase instead of FE?
>>
>> VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
>>
>> For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
>>
>> ***During gimplification,
>> For a call to __builtin_dynamic_object_size (ptr, type)
>> Check whether the type of ptr includes counted_by attribute, if so, change the call to
>> __builtin_dynamic_object_size (ptr, type, counted_by field)
>>
>> Then the correct data dependence should be represented well in the IR.
>>
>> **During object size phase,
>>
>> The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
>
> But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
Yes, this is true.
I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
Joseph, if implement this in FE, where in the FE I should look at?
Thanks a lot for the help.
Qing
> For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
>
> Richard
>
>>
>>>
>>> A related issue is that assignment to the field and storage allocation
>>> are not tied together
>>
>> Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
>>
>> For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
>>
>>> - if there's no use of the size data we might
>>> remove the store of it as dead.
>>
>> Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
>> I guess that the compiler should be able to do this automatically?
>>
>> thanks.
>>
>> Qing
>>>
>>> Of course I guess __bos then behaves like sizeof ().
>>>
>>> Richard.
>>>
>>>>
>>>> Qing
>>>>
>>>>>
>>>>> It may not work for something like this though:
>>>>>
>>>>> static size_t
>>>>> get_size_of (void *ptr)
>>>>> {
>>>>> return __bdos (ptr, 1);
>>>>> }
>>>>>
>>>>> void
>>>>> foo (size_t sz)
>>>>> {
>>>>> array_annotated = __builtin_malloc (sz);
>>>>> array_annotated = sz;
>>>>>
>>>>> ...
>>>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>>>> ...
>>>>> }
>>>>>
>>>>> because the call to get_size_of () may not have been inlined that early.
>>>>>
>>>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>>>
>>>>> Sid
>>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 16:37 ` Qing Zhao
@ 2023-10-23 18:06 ` Martin Uecker
2023-10-23 18:31 ` Martin Uecker
` (2 more replies)
2023-10-23 18:10 ` Joseph Myers
1 sibling, 3 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-23 18:06 UTC (permalink / raw)
To: Qing Zhao, Richard Biener, Joseph Myers
Cc: Siddhesh Poyarekar, Jakub Jelinek, gcc Patches, kees Cook, isanbard
Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
>
> > On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> >
> >
> >
> > > Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
> > >
> > >
> > >
> > > > On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > > >
> > > > > On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
> > > > >
> > > > >
> > > > >
> > > > > > On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
> > > > > >
> > > > > > On 2023-10-20 14:38, Qing Zhao wrote:
> > > > > > > How about the following:
> > > > > > > Add one more parameter to __builtin_dynamic_object_size(), i.e
> > > > > > > __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
> > > > > > > When we see the structure field has counted_by attribute.
> > > > > >
> > > > > > Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
> > > > >
> > > > > Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
> > > > >
> > > > > But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
> > > > >
> > > > > This might be the simplest solution?
> > > >
> > > > If the dynamic object size is derived of a field then I think you need to
> > > > put the "load" of that memory location at the point (as argument)
> > > > of the __bos call right at parsing time. I know that's awkward because
> > > > you try to play tricks "discovering" that field only late, but that's not
> > > > going to work.
> > >
> > > Is it better to do this at gimplification phase instead of FE?
> > >
> > > VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
> > >
> > > For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
> > >
> > > ***During gimplification,
> > > For a call to __builtin_dynamic_object_size (ptr, type)
> > > Check whether the type of ptr includes counted_by attribute, if so, change the call to
> > > __builtin_dynamic_object_size (ptr, type, counted_by field)
> > >
> > > Then the correct data dependence should be represented well in the IR.
> > >
> > > **During object size phase,
> > >
> > > The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
> >
> > But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
>
> Yes, this is true.
> I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
>
> Joseph, if implement this in FE, where in the FE I should look at?
>
We should aim for a good integration with the BDOS pass, so
that it can propagate the information further, e.g. the
following should work:
struct { int L; char buf[] __counted_by(L) } x;
x.L = N;
x.buf = ...;
char *p = &x->f;
__bdos(p) -> N
So we need to be smart on how we provide the size
information for x->f to the backend.
This would also be desirable for the language extension.
Martin
> Thanks a lot for the help.
>
> Qing
>
> > For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
> >
> > Richard
> >
> > >
> > > >
> > > > A related issue is that assignment to the field and storage allocation
> > > > are not tied together
> > >
> > > Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
> > >
> > > For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
> > >
> > > > - if there's no use of the size data we might
> > > > remove the store of it as dead.
> > >
> > > Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
> > > I guess that the compiler should be able to do this automatically?
> > >
> > > thanks.
> > >
> > > Qing
> > > >
> > > > Of course I guess __bos then behaves like sizeof ().
> > > >
> > > > Richard.
> > > >
> > > > >
> > > > > Qing
> > > > >
> > > > > >
> > > > > > It may not work for something like this though:
> > > > > >
> > > > > > static size_t
> > > > > > get_size_of (void *ptr)
> > > > > > {
> > > > > > return __bdos (ptr, 1);
> > > > > > }
> > > > > >
> > > > > > void
> > > > > > foo (size_t sz)
> > > > > > {
> > > > > > array_annotated = __builtin_malloc (sz);
> > > > > > array_annotated = sz;
> > > > > >
> > > > > > ...
> > > > > > __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
> > > > > > ...
> > > > > > }
> > > > > >
> > > > > > because the call to get_size_of () may not have been inlined that early.
> > > > > >
> > > > > > The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
> > > > > >
> > > > > > Sid
> > >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 16:37 ` Qing Zhao
2023-10-23 18:06 ` Martin Uecker
@ 2023-10-23 18:10 ` Joseph Myers
1 sibling, 0 replies; 116+ messages in thread
From: Joseph Myers @ 2023-10-23 18:10 UTC (permalink / raw)
To: Qing Zhao
Cc: Richard Biener, Siddhesh Poyarekar, Jakub Jelinek, gcc Patches,
kees Cook, Martin Uecker, isanbard
On Mon, 23 Oct 2023, Qing Zhao wrote:
> I prefer to implement this in gimplification phase since I am more
> familiar with the code there.. (I think that implementing it in
> gimplification should be very similar as implementing it in FE? Or do I
> miss anything here?)
>
> Joseph, if implement this in FE, where in the FE I should look at?
I tend to think that gimplification time is appropriate for adding this
dependency, but if you wish to rewrite a built-in function call in the
front end before then, it could be done in build_function_call_vec.
--
Joseph S. Myers
joseph@codesourcery.com
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 19:54 ` Martin Uecker
@ 2023-10-23 18:17 ` Qing Zhao
2023-10-23 19:52 ` Kees Cook
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 18:17 UTC (permalink / raw)
To: Martin Uecker; +Cc: Kees Cook, gcc-patches
> On Oct 20, 2023, at 3:54 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Freitag, dem 20.10.2023 um 18:48 +0000 schrieb Qing Zhao:
>>
>>> On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
>>>
>>> On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
>>>> Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
>>>>> On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
>>>>>> As I replied to Martin in another email, I plan to do the following to resolve this issue:
>>>>>>
>>>>>> 1. No specification for signed or unsigned for counted_by field.
>>>>>> 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
>>>>>
>>>>> I don't understand why this needs to be a runtime sanitizer. The
>>>>> signedness is known at compile time, so I would expect a -W option.
>>>>
>>>> The signedness of the type but not of the value.
>>>>
>>>> But I would not want to have a warning for signed
>>>> counter types by default because I would prefer
>>>> to use signed types (for various reasons including
>>>> better overflow detection).
>>>>
>>>>> Or
>>>>> do you mean you'd split up -fsanitize=bounds between unsigned and signed
>>>>> indexes? I'd find that kind of awkward for the kernel... but I feel like
>>>>> I've misunderstood something. :)
>>>>>
>>>>> -Kees
>>>>
>>>> The idea would be to detect at run-time the case
>>>> if x->buf is used at a time where x->counter
>>>> is negative and also when x->counter * sizeof(x->buf[0])
>>>> overflows or is too big.
>>>>
>>>> This would be similar to
>>>>
>>>> int a[n];
>>>>
>>>> where it is detected at run-time if n is not-positive.
>>>
>>> Right. I guess what I mean to say is that I would expect this case to
>>> already be caught by -fsanitize=bounds -- I don't see a reason to add an
>>> additional sanitizer option.
>>>
>>> struct foo {
>>> int count;
>>> int array[] __counted_by(count);
>>> };
>>>
>>> foo->count = 5;
>>> foo->array[0] = 1; // ok
>>> foo->array[10] = 1; // -fsanitize=bounds will catch this
>>> foo->array[-10] = 1; // -fsanitize=bounds will catch this too
>>>
>>>
>>
>> just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
>>
>> ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
>> ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
>>
>
> Yes, but I thought we were discussing the case where count is
> set to a negative value:
>
> foo->count = -1;
> int x = foo->array[3]; // UBSan should diagnose this
>
> And also the case when foo->array becomes too big.
Oops, yes, you are right.
Thanks.
Qing
>
> Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:06 ` Martin Uecker
@ 2023-10-23 18:31 ` Martin Uecker
2023-10-23 19:00 ` Qing Zhao
2023-10-23 18:33 ` Qing Zhao
2023-10-23 18:43 ` Siddhesh Poyarekar
2 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-23 18:31 UTC (permalink / raw)
To: Qing Zhao, Richard Biener, Joseph Myers
Cc: Siddhesh Poyarekar, Jakub Jelinek, gcc Patches, kees Cook, isanbard
Am Montag, dem 23.10.2023 um 20:06 +0200 schrieb Martin Uecker:
> Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
> >
> > > On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > >
> > >
> > >
> > > > Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
> > > >
> > > >
> > > >
> > > > > On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > > > >
> > > > > > On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
> > > > > >
> > > > > >
> > > > > >
> > > > > > > On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
> > > > > > >
> > > > > > > On 2023-10-20 14:38, Qing Zhao wrote:
> > > > > > > > How about the following:
> > > > > > > > Add one more parameter to __builtin_dynamic_object_size(), i.e
> > > > > > > > __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
> > > > > > > > When we see the structure field has counted_by attribute.
> > > > > > >
> > > > > > > Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
> > > > > >
> > > > > > Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
> > > > > >
> > > > > > But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
> > > > > >
> > > > > > This might be the simplest solution?
> > > > >
> > > > > If the dynamic object size is derived of a field then I think you need to
> > > > > put the "load" of that memory location at the point (as argument)
> > > > > of the __bos call right at parsing time. I know that's awkward because
> > > > > you try to play tricks "discovering" that field only late, but that's not
> > > > > going to work.
> > > >
> > > > Is it better to do this at gimplification phase instead of FE?
> > > >
> > > > VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
> > > >
> > > > For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
> > > >
> > > > ***During gimplification,
> > > > For a call to __builtin_dynamic_object_size (ptr, type)
> > > > Check whether the type of ptr includes counted_by attribute, if so, change the call to
> > > > __builtin_dynamic_object_size (ptr, type, counted_by field)
> > > >
> > > > Then the correct data dependence should be represented well in the IR.
> > > >
> > > > **During object size phase,
> > > >
> > > > The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
> > >
> > > But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
> >
> > Yes, this is true.
> > I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
> >
> > Joseph, if implement this in FE, where in the FE I should look at?
> >
>
> We should aim for a good integration with the BDOS pass, so
> that it can propagate the information further, e.g. the
> following should work:
>
> struct { int L; char buf[] __counted_by(L) } x;
> x.L = N;
> x.buf = ...;
> char *p = &x->f;
> __bdos(p) -> N
>
> So we need to be smart on how we provide the size
> information for x->f to the backend.
To follow up on this. I do not think we should change the
builtin in the FE or gimplification. Instead, we want
to change the field access and compute the size there.
In my toy patch I then made this have a VLA type that
encodes the size. Here, this would need to be done
differently.
But still, what we are missing in both cases
is a proper way to pass the information down to BDOS.
For VLAs this works because BDOS can see the size of
the definition. For calls to allocation functions
it is read from an attribute.
But I am not sure what would be the best way to encode
this information so that BDOS can later access it.
Martin
>
> This would also be desirable for the language extension.
>
> Martin
>
>
> > Thanks a lot for the help.
> >
> > Qing
> >
> > > For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
> > >
> > > Richard
> > >
> > > >
> > > > >
> > > > > A related issue is that assignment to the field and storage allocation
> > > > > are not tied together
> > > >
> > > > Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
> > > >
> > > > For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
> > > >
> > > > > - if there's no use of the size data we might
> > > > > remove the store of it as dead.
> > > >
> > > > Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
> > > > I guess that the compiler should be able to do this automatically?
> > > >
> > > > thanks.
> > > >
> > > > Qing
> > > > >
> > > > > Of course I guess __bos then behaves like sizeof ().
> > > > >
> > > > > Richard.
> > > > >
> > > > > >
> > > > > > Qing
> > > > > >
> > > > > > >
> > > > > > > It may not work for something like this though:
> > > > > > >
> > > > > > > static size_t
> > > > > > > get_size_of (void *ptr)
> > > > > > > {
> > > > > > > return __bdos (ptr, 1);
> > > > > > > }
> > > > > > >
> > > > > > > void
> > > > > > > foo (size_t sz)
> > > > > > > {
> > > > > > > array_annotated = __builtin_malloc (sz);
> > > > > > > array_annotated = sz;
> > > > > > >
> > > > > > > ...
> > > > > > > __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
> > > > > > > ...
> > > > > > > }
> > > > > > >
> > > > > > > because the call to get_size_of () may not have been inlined that early.
> > > > > > >
> > > > > > > The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
> > > > > > >
> > > > > > > Sid
> > > >
> >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:06 ` Martin Uecker
2023-10-23 18:31 ` Martin Uecker
@ 2023-10-23 18:33 ` Qing Zhao
2023-10-23 18:43 ` Siddhesh Poyarekar
2 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 18:33 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Joseph Myers, Siddhesh Poyarekar, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 23, 2023, at 2:06 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
>>
>>> On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>
>>>
>>>
>>>> Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
>>>>
>>>>
>>>>
>>>>> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>>>
>>>>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>>>>
>>>>>>
>>>>>>
>>>>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>>>>
>>>>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>>>>> How about the following:
>>>>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>>>>> When we see the structure field has counted_by attribute.
>>>>>>>
>>>>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>>>>
>>>>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>>>>
>>>>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>>>>
>>>>>> This might be the simplest solution?
>>>>>
>>>>> If the dynamic object size is derived of a field then I think you need to
>>>>> put the "load" of that memory location at the point (as argument)
>>>>> of the __bos call right at parsing time. I know that's awkward because
>>>>> you try to play tricks "discovering" that field only late, but that's not
>>>>> going to work.
>>>>
>>>> Is it better to do this at gimplification phase instead of FE?
>>>>
>>>> VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
>>>>
>>>> For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
>>>>
>>>> ***During gimplification,
>>>> For a call to __builtin_dynamic_object_size (ptr, type)
>>>> Check whether the type of ptr includes counted_by attribute, if so, change the call to
>>>> __builtin_dynamic_object_size (ptr, type, counted_by field)
>>>>
>>>> Then the correct data dependence should be represented well in the IR.
>>>>
>>>> **During object size phase,
>>>>
>>>> The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
>>>
>>> But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
>>
>> Yes, this is true.
>> I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
>>
>> Joseph, if implement this in FE, where in the FE I should look at?
>>
>
> We should aim for a good integration with the BDOS pass, so
> that it can propagate the information further, e.g. the
> following should work:
>
> struct { int L; char buf[] __counted_by(L) } x;
> x.L = N;
> x.buf = ...;
> char *p = &x->f;
Is the above line should be:
char *p = &x.buf
?
> __bdos(p) -> N
>
> So we need to be smart on how we provide the size
> information for x->f to the backend.
Do you have any other suggestion here?
(Right now, what we’d like to do is to add one more argument for the function __bdos as
__bdos (p, type, x.L))
>
> This would also be desirable for the language extension.
Yes.
Qing
>
> Martin
>
>
>> Thanks a lot for the help.
>>
>> Qing
>>
>>> For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
>>>
>>> Richard
>>>
>>>>
>>>>>
>>>>> A related issue is that assignment to the field and storage allocation
>>>>> are not tied together
>>>>
>>>> Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
>>>>
>>>> For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
>>>>
>>>>> - if there's no use of the size data we might
>>>>> remove the store of it as dead.
>>>>
>>>> Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
>>>> I guess that the compiler should be able to do this automatically?
>>>>
>>>> thanks.
>>>>
>>>> Qing
>>>>>
>>>>> Of course I guess __bos then behaves like sizeof ().
>>>>>
>>>>> Richard.
>>>>>
>>>>>>
>>>>>> Qing
>>>>>>
>>>>>>>
>>>>>>> It may not work for something like this though:
>>>>>>>
>>>>>>> static size_t
>>>>>>> get_size_of (void *ptr)
>>>>>>> {
>>>>>>> return __bdos (ptr, 1);
>>>>>>> }
>>>>>>>
>>>>>>> void
>>>>>>> foo (size_t sz)
>>>>>>> {
>>>>>>> array_annotated = __builtin_malloc (sz);
>>>>>>> array_annotated = sz;
>>>>>>>
>>>>>>> ...
>>>>>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>>>>>> ...
>>>>>>> }
>>>>>>>
>>>>>>> because the call to get_size_of () may not have been inlined that early.
>>>>>>>
>>>>>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>>>>>
>>>>>>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:06 ` Martin Uecker
2023-10-23 18:31 ` Martin Uecker
2023-10-23 18:33 ` Qing Zhao
@ 2023-10-23 18:43 ` Siddhesh Poyarekar
2023-10-23 18:55 ` Martin Uecker
2023-10-23 19:43 ` Qing Zhao
2 siblings, 2 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-23 18:43 UTC (permalink / raw)
To: Martin Uecker, Qing Zhao, Richard Biener, Joseph Myers
Cc: Jakub Jelinek, gcc Patches, kees Cook, isanbard
On 2023-10-23 14:06, Martin Uecker wrote:
> We should aim for a good integration with the BDOS pass, so
> that it can propagate the information further, e.g. the
> following should work:
>
> struct { int L; char buf[] __counted_by(L) } x;
> x.L = N;
> x.buf = ...;
> char *p = &x->f;
> __bdos(p) -> N
>
> So we need to be smart on how we provide the size
> information for x->f to the backend.
>
> This would also be desirable for the language extension.
This is essentially why there need to be frontend rules constraining
reordering and reachability semantics of x.L, thus restricting DSE and
reordering for it. This is not really a __bdos/__bos question, because
that bit is trivial; if the structure is visible, the value is simply
x.L. This is also why adding a reference to x.L in __bos/__bdos is not
sufficient or even possible in, e.g. the above case you note.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:43 ` Siddhesh Poyarekar
@ 2023-10-23 18:55 ` Martin Uecker
2023-10-23 19:43 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-23 18:55 UTC (permalink / raw)
To: Siddhesh Poyarekar, Qing Zhao, Richard Biener, Joseph Myers
Cc: Jakub Jelinek, gcc Patches, kees Cook, isanbard
Am Montag, dem 23.10.2023 um 14:43 -0400 schrieb Siddhesh Poyarekar:
> On 2023-10-23 14:06, Martin Uecker wrote:
> > We should aim for a good integration with the BDOS pass, so
> > that it can propagate the information further, e.g. the
> > following should work:
> >
> > struct { int L; char buf[] __counted_by(L) } x;
> > x.L = N;
> > x.buf = ...;
> > char *p = &x->f;
> > __bdos(p) -> N
> >
> > So we need to be smart on how we provide the size
> > information for x->f to the backend.
> >
> > This would also be desirable for the language extension.
>
> This is essentially why there need to be frontend rules constraining
> reordering and reachability semantics of x.L, thus restricting DSE and
> reordering for it.
Yes, this too.
> This is not really a __bdos/__bos question, because
> that bit is trivial; if the structure is visible, the value is simply
> x.L. This is also why adding a reference to x.L in __bos/__bdos is not
> sufficient or even possible in, e.g. the above case you note.
The value x.L may change in time. I would argue that it needs
to be the value of x.L at the time where x.buf (not x->f, sorry)
is accessed. So the FE needs to evaluate x.L when x.buf is
accessed and store the value somewhere where __bdos can find
it later. In the type information would make sense.
But I am not sure how to do this in the best way so that this
information is not removed later when not used explicitely
before __bdos tries to look at it.
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:31 ` Martin Uecker
@ 2023-10-23 19:00 ` Qing Zhao
2023-10-23 19:37 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 19:00 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Joseph Myers, Siddhesh Poyarekar, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 23, 2023, at 2:31 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Montag, dem 23.10.2023 um 20:06 +0200 schrieb Martin Uecker:
>> Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
>>>
>>>> On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>>
>>>>
>>>>
>>>>> Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
>>>>>
>>>>>
>>>>>
>>>>>> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>>>>
>>>>>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>>>>>
>>>>>>>
>>>>>>>
>>>>>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>>>>>
>>>>>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>>>>>> How about the following:
>>>>>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>>>>>> When we see the structure field has counted_by attribute.
>>>>>>>>
>>>>>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>>>>>
>>>>>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>>>>>
>>>>>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>>>>>
>>>>>>> This might be the simplest solution?
>>>>>>
>>>>>> If the dynamic object size is derived of a field then I think you need to
>>>>>> put the "load" of that memory location at the point (as argument)
>>>>>> of the __bos call right at parsing time. I know that's awkward because
>>>>>> you try to play tricks "discovering" that field only late, but that's not
>>>>>> going to work.
>>>>>
>>>>> Is it better to do this at gimplification phase instead of FE?
>>>>>
>>>>> VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
>>>>>
>>>>> For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
>>>>>
>>>>> ***During gimplification,
>>>>> For a call to __builtin_dynamic_object_size (ptr, type)
>>>>> Check whether the type of ptr includes counted_by attribute, if so, change the call to
>>>>> __builtin_dynamic_object_size (ptr, type, counted_by field)
>>>>>
>>>>> Then the correct data dependence should be represented well in the IR.
>>>>>
>>>>> **During object size phase,
>>>>>
>>>>> The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
>>>>
>>>> But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
>>>
>>> Yes, this is true.
>>> I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
>>>
>>> Joseph, if implement this in FE, where in the FE I should look at?
>>>
>>
>> We should aim for a good integration with the BDOS pass, so
>> that it can propagate the information further, e.g. the
>> following should work:
>>
>> struct { int L; char buf[] __counted_by(L) } x;
>> x.L = N;
>> x.buf = ...;
>> char *p = &x->f;
>> __bdos(p) -> N
>>
>> So we need to be smart on how we provide the size
>> information for x->f to the backend.
>
> To follow up on this. I do not think we should change the
> builtin in the FE or gimplification. Instead, we want
> to change the field access and compute the size there.
Could you please clarify on this? What do you mean by "change the field access and compute the size there”?
>
> In my toy patch I then made this have a VLA type that
> encodes the size. Here, this would need to be done
> differently.
>
> But still, what we are missing in both cases
> is a proper way to pass the information down to BDOS.
What’ s the issue with adding a new argument (x.L) to the BDOS call? What’s missing with this approach?
>
> For VLAs this works because BDOS can see the size of
> the definition. For calls to allocation functions
> it is read from an attribute.
You mean for VLA, BDOS see the size of the definition from the attribute for the allocation function?
Yes, that’s the case for VLA.
For VLA, the size computation and storage allocation are all done by the compiler (through “gimplify_vla_decl” in gimplification phase),
So these two can be tied together by the compiler.
However, for FMA with counted_by attribute, the storage allocation and the counted_by assignment are done by the user.
Qing
>
> But I am not sure what would be the best way to encode
> this information so that BDOS can later access it.
>
> Martin
>
>
>
>
>>
>> This would also be desirable for the language extension.
>>
>> Martin
>>
>>
>>> Thanks a lot for the help.
>>>
>>> Qing
>>>
>>>> For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
>>>>
>>>> Richard
>>>>
>>>>>
>>>>>>
>>>>>> A related issue is that assignment to the field and storage allocation
>>>>>> are not tied together
>>>>>
>>>>> Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
>>>>>
>>>>> For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
>>>>>
>>>>>> - if there's no use of the size data we might
>>>>>> remove the store of it as dead.
>>>>>
>>>>> Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
>>>>> I guess that the compiler should be able to do this automatically?
>>>>>
>>>>> thanks.
>>>>>
>>>>> Qing
>>>>>>
>>>>>> Of course I guess __bos then behaves like sizeof ().
>>>>>>
>>>>>> Richard.
>>>>>>
>>>>>>>
>>>>>>> Qing
>>>>>>>
>>>>>>>>
>>>>>>>> It may not work for something like this though:
>>>>>>>>
>>>>>>>> static size_t
>>>>>>>> get_size_of (void *ptr)
>>>>>>>> {
>>>>>>>> return __bdos (ptr, 1);
>>>>>>>> }
>>>>>>>>
>>>>>>>> void
>>>>>>>> foo (size_t sz)
>>>>>>>> {
>>>>>>>> array_annotated = __builtin_malloc (sz);
>>>>>>>> array_annotated = sz;
>>>>>>>>
>>>>>>>> ...
>>>>>>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>>>>>>> ...
>>>>>>>> }
>>>>>>>>
>>>>>>>> because the call to get_size_of () may not have been inlined that early.
>>>>>>>>
>>>>>>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>>>>>>
>>>>>>>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 19:00 ` Qing Zhao
@ 2023-10-23 19:37 ` Martin Uecker
2023-10-23 20:33 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-23 19:37 UTC (permalink / raw)
To: Qing Zhao
Cc: Richard Biener, Joseph Myers, Siddhesh Poyarekar, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Montag, dem 23.10.2023 um 19:00 +0000 schrieb Qing Zhao:
>
> > On Oct 23, 2023, at 2:31 PM, Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Montag, dem 23.10.2023 um 20:06 +0200 schrieb Martin Uecker:
> > > Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
> > > >
> > > > > On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > > > >
> > > > >
> > > > >
> > > > > > Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
> > > > > >
> > > > > >
> > > > > >
> > > > > > > On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > > > > > >
> > > > > > > > On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
> > > > > > > >
> > > > > > > >
> > > > > > > >
> > > > > > > > > On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
> > > > > > > > >
> > > > > > > > > On 2023-10-20 14:38, Qing Zhao wrote:
> > > > > > > > > > How about the following:
> > > > > > > > > > Add one more parameter to __builtin_dynamic_object_size(), i.e
> > > > > > > > > > __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
> > > > > > > > > > When we see the structure field has counted_by attribute.
> > > > > > > > >
> > > > > > > > > Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
> > > > > > > >
> > > > > > > > Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
> > > > > > > >
> > > > > > > > But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
> > > > > > > >
> > > > > > > > This might be the simplest solution?
> > > > > > >
> > > > > > > If the dynamic object size is derived of a field then I think you need to
> > > > > > > put the "load" of that memory location at the point (as argument)
> > > > > > > of the __bos call right at parsing time. I know that's awkward because
> > > > > > > you try to play tricks "discovering" that field only late, but that's not
> > > > > > > going to work.
> > > > > >
> > > > > > Is it better to do this at gimplification phase instead of FE?
> > > > > >
> > > > > > VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
> > > > > >
> > > > > > For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
> > > > > >
> > > > > > ***During gimplification,
> > > > > > For a call to __builtin_dynamic_object_size (ptr, type)
> > > > > > Check whether the type of ptr includes counted_by attribute, if so, change the call to
> > > > > > __builtin_dynamic_object_size (ptr, type, counted_by field)
> > > > > >
> > > > > > Then the correct data dependence should be represented well in the IR.
> > > > > >
> > > > > > **During object size phase,
> > > > > >
> > > > > > The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
> > > > >
> > > > > But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
> > > >
> > > > Yes, this is true.
> > > > I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
> > > >
> > > > Joseph, if implement this in FE, where in the FE I should look at?
> > > >
> > >
> > > We should aim for a good integration with the BDOS pass, so
> > > that it can propagate the information further, e.g. the
> > > following should work:
> > >
> > > struct { int L; char buf[] __counted_by(L) } x;
> > > x.L = N;
> > > x.buf = ...;
> > > char *p = &x->f;
> > > __bdos(p) -> N
> > >
> > > So we need to be smart on how we provide the size
> > > information for x->f to the backend.
> >
> > To follow up on this. I do not think we should change the
> > builtin in the FE or gimplification. Instead, we want
> > to change the field access and compute the size there.
> Could you please clarify on this? What do you mean by
> "change the field access and compute the size there”?
I think the FE should essentially give the
type
char [buf.L]
to buf.x;
If the type (or its size) could be preserved
at this point so that it can be later
discovered by __bdos, then it could know
the size and propagate it further.
For the attribute, this is not exactly what
the FE could do because the semantic type
can not change, but this is roughly the idea.
> >
> > In my toy patch I then made this have a VLA type that
> > encodes the size. Here, this would need to be done
> > differently.
> >
> > But still, what we are missing in both cases
> > is a proper way to pass the information down to BDOS.
>
> What’ s the issue with adding a new argument (x.L) to the BDOS call? What’s missing with this approach?
>
See the example above. the BDOS call might come much
later when the relationship of the pointer to the
field access is no longer there.
> >
> > For VLAs this works because BDOS can see the size of
> > the definition. For calls to allocation functions
> > it is read from an attribute.
>
> You mean for VLA, BDOS see the size of the definition
> from the attribute for the allocation function?
> Yes, that’s the case for VLA.
Ok, I am wrong about how it works for VLAs. They
get transformed to an alloca.
But all calls marked with alloc_size and other
allocations functions are detected in BDOS.
>
> For VLA, the size computation and storage allocation are all done by the compiler (through “gimplify_vla_decl” in gimplification phase),
> So these two can be tied together by the compiler.
>
> However, for FMA with counted_by attribute, the
> storage allocation and the counted_by assignment
> are done by the user.
Yes.
Martin
>
> Qing
> >
> > But I am not sure what would be the best way to encode
> > this information so that BDOS can later access it.
> >
> > Martin
> >
> >
> >
> >
> > >
> > > This would also be desirable for the language extension.
> > >
> > > Martin
> > >
> > >
> > > > Thanks a lot for the help.
> > > >
> > > > Qing
> > > >
> > > > > For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
> > > > >
> > > > > Richard
> > > > >
> > > > > >
> > > > > > >
> > > > > > > A related issue is that assignment to the field and storage allocation
> > > > > > > are not tied together
> > > > > >
> > > > > > Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
> > > > > >
> > > > > > For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
> > > > > >
> > > > > > > - if there's no use of the size data we might
> > > > > > > remove the store of it as dead.
> > > > > >
> > > > > > Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
> > > > > > I guess that the compiler should be able to do this automatically?
> > > > > >
> > > > > > thanks.
> > > > > >
> > > > > > Qing
> > > > > > >
> > > > > > > Of course I guess __bos then behaves like sizeof ().
> > > > > > >
> > > > > > > Richard.
> > > > > > >
> > > > > > > >
> > > > > > > > Qing
> > > > > > > >
> > > > > > > > >
> > > > > > > > > It may not work for something like this though:
> > > > > > > > >
> > > > > > > > > static size_t
> > > > > > > > > get_size_of (void *ptr)
> > > > > > > > > {
> > > > > > > > > return __bdos (ptr, 1);
> > > > > > > > > }
> > > > > > > > >
> > > > > > > > > void
> > > > > > > > > foo (size_t sz)
> > > > > > > > > {
> > > > > > > > > array_annotated = __builtin_malloc (sz);
> > > > > > > > > array_annotated = sz;
> > > > > > > > >
> > > > > > > > > ...
> > > > > > > > > __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
> > > > > > > > > ...
> > > > > > > > > }
> > > > > > > > >
> > > > > > > > > because the call to get_size_of () may not have been inlined that early.
> > > > > > > > >
> > > > > > > > > The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
> > > > > > > > >
> > > > > > > > > Sid
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 18:43 ` Siddhesh Poyarekar
2023-10-23 18:55 ` Martin Uecker
@ 2023-10-23 19:43 ` Qing Zhao
2023-10-23 22:48 ` Siddhesh Poyarekar
1 sibling, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 19:43 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Martin Uecker, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 23, 2023, at 2:43 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-23 14:06, Martin Uecker wrote:
>> We should aim for a good integration with the BDOS pass, so
>> that it can propagate the information further, e.g. the
>> following should work:
>> struct { int L; char buf[] __counted_by(L) } x;
>> x.L = N;
>> x.buf = ...;
>> char *p = &x->f;
>> __bdos(p) -> N
>> So we need to be smart on how we provide the size
>> information for x->f to the backend.
>> This would also be desirable for the language extension.
>
> This is essentially why there need to be frontend rules constraining reordering and reachability semantics of x.L, thus restricting DSE and reordering for it.
My understanding is that Restricting DSE and reordering should be done by the proper data flow information, with a new argument added to the BDOS call, this correct data flow information could be maintained, and then the DSE and reordering will not happen.
I don’t quite understand what kind of frontend rules should be added to constrain reordering and reachability semantics? Can you explain this a little bit more? Do you mean to add some rules or requirment to the new attribute that the users of the attribute should follow in the source code?
> This is not really a __bdos/__bos question, because that bit is trivial; if the structure is visible, the value is simply x.L. This is also why adding a reference to x.L in __bos/__bdos is not sufficient or even possible in, e.g. the above case you note.
I am a little confused here, are we discussing how to resolve the potential reordering issue of the following:
"
struct annotated {
size_t foo;
char array[] __attribute__((counted_by (foo)));
};
p->foo = 10;
size = __builtin_dynamic_object_size (p->array,1);
“?
Or a bigger issue?
Qing
>
> Thanks,
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-20 19:54 ` Martin Uecker
2023-10-23 18:17 ` Qing Zhao
@ 2023-10-23 19:52 ` Kees Cook
2023-10-23 19:57 ` Martin Uecker
1 sibling, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-23 19:52 UTC (permalink / raw)
To: Martin Uecker; +Cc: Qing Zhao, gcc-patches
On Fri, Oct 20, 2023 at 09:54:05PM +0200, Martin Uecker wrote:
> Am Freitag, dem 20.10.2023 um 18:48 +0000 schrieb Qing Zhao:
> >
> > > On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
> > >
> > > On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
> > > > Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> > > > > On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > > > > > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> > > > > >
> > > > > > 1. No specification for signed or unsigned for counted_by field.
> > > > > > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
> > > > >
> > > > > I don't understand why this needs to be a runtime sanitizer. The
> > > > > signedness is known at compile time, so I would expect a -W option.
> > > >
> > > > The signedness of the type but not of the value.
> > > >
> > > > But I would not want to have a warning for signed
> > > > counter types by default because I would prefer
> > > > to use signed types (for various reasons including
> > > > better overflow detection).
> > > >
> > > > > Or
> > > > > do you mean you'd split up -fsanitize=bounds between unsigned and signed
> > > > > indexes? I'd find that kind of awkward for the kernel... but I feel like
> > > > > I've misunderstood something. :)
> > > > >
> > > > > -Kees
> > > >
> > > > The idea would be to detect at run-time the case
> > > > if x->buf is used at a time where x->counter
> > > > is negative and also when x->counter * sizeof(x->buf[0])
> > > > overflows or is too big.
> > > >
> > > > This would be similar to
> > > >
> > > > int a[n];
> > > >
> > > > where it is detected at run-time if n is not-positive.
> > >
> > > Right. I guess what I mean to say is that I would expect this case to
> > > already be caught by -fsanitize=bounds -- I don't see a reason to add an
> > > additional sanitizer option.
> > >
> > > struct foo {
> > > int count;
> > > int array[] __counted_by(count);
> > > };
> > >
> > > foo->count = 5;
> > > foo->array[0] = 1; // ok
> > > foo->array[10] = 1; // -fsanitize=bounds will catch this
> > > foo->array[-10] = 1; // -fsanitize=bounds will catch this too
> > >
> > >
> >
> > just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
> >
> > ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
> > ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
> >
>
> Yes, but I thought we were discussing the case where count is
> set to a negative value:
>
> foo->count = -1;
> int x = foo->array[3]; // UBSan should diagnose this
Oh right, I keep thinking about it backwards.
Yeah, we can't trap the "count" assignment, because it may be getting used
for other purposes. But yeah, access to "array" should trap if "count"
is negative.
> And also the case when foo->array becomes too big.
How do you mean?
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 19:52 ` Kees Cook
@ 2023-10-23 19:57 ` Martin Uecker
2023-10-23 22:03 ` Kees Cook
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-23 19:57 UTC (permalink / raw)
To: Kees Cook; +Cc: Qing Zhao, gcc-patches
Am Montag, dem 23.10.2023 um 12:52 -0700 schrieb Kees Cook:
> On Fri, Oct 20, 2023 at 09:54:05PM +0200, Martin Uecker wrote:
> > Am Freitag, dem 20.10.2023 um 18:48 +0000 schrieb Qing Zhao:
> > >
> > > > On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
> > > >
> > > > On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
> > > > > Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> > > > > > On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > > > > > > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> > > > > > >
> > > > > > > 1. No specification for signed or unsigned for counted_by field.
> > > > > > > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
> > > > > >
> > > > > > I don't understand why this needs to be a runtime sanitizer. The
> > > > > > signedness is known at compile time, so I would expect a -W option.
> > > > >
> > > > > The signedness of the type but not of the value.
> > > > >
> > > > > But I would not want to have a warning for signed
> > > > > counter types by default because I would prefer
> > > > > to use signed types (for various reasons including
> > > > > better overflow detection).
> > > > >
> > > > > > Or
> > > > > > do you mean you'd split up -fsanitize=bounds between unsigned and signed
> > > > > > indexes? I'd find that kind of awkward for the kernel... but I feel like
> > > > > > I've misunderstood something. :)
> > > > > >
> > > > > > -Kees
> > > > >
> > > > > The idea would be to detect at run-time the case
> > > > > if x->buf is used at a time where x->counter
> > > > > is negative and also when x->counter * sizeof(x->buf[0])
> > > > > overflows or is too big.
> > > > >
> > > > > This would be similar to
> > > > >
> > > > > int a[n];
> > > > >
> > > > > where it is detected at run-time if n is not-positive.
> > > >
> > > > Right. I guess what I mean to say is that I would expect this case to
> > > > already be caught by -fsanitize=bounds -- I don't see a reason to add an
> > > > additional sanitizer option.
> > > >
> > > > struct foo {
> > > > int count;
> > > > int array[] __counted_by(count);
> > > > };
> > > >
> > > > foo->count = 5;
> > > > foo->array[0] = 1; // ok
> > > > foo->array[10] = 1; // -fsanitize=bounds will catch this
> > > > foo->array[-10] = 1; // -fsanitize=bounds will catch this too
> > > >
> > > >
> > >
> > > just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
> > >
> > > ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
> > > ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
> > >
> >
> > Yes, but I thought we were discussing the case where count is
> > set to a negative value:
> >
> > foo->count = -1;
> > int x = foo->array[3]; // UBSan should diagnose this
>
> Oh right, I keep thinking about it backwards.
>
> Yeah, we can't trap the "count" assignment, because it may be getting used
> for other purposes. But yeah, access to "array" should trap if "count"
> is negative.
>
> > And also the case when foo->array becomes too big.
>
> How do you mean?
count * sizeof(member) could overflow or otherwise be
bigger than allowed.
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 19:37 ` Martin Uecker
@ 2023-10-23 20:33 ` Qing Zhao
0 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-23 20:33 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Joseph Myers, Siddhesh Poyarekar, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 23, 2023, at 3:37 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Montag, dem 23.10.2023 um 19:00 +0000 schrieb Qing Zhao:
>>
>>> On Oct 23, 2023, at 2:31 PM, Martin Uecker <uecker@tugraz.at> wrote:
>>>
>>> Am Montag, dem 23.10.2023 um 20:06 +0200 schrieb Martin Uecker:
>>>> Am Montag, dem 23.10.2023 um 16:37 +0000 schrieb Qing Zhao:
>>>>>
>>>>>> On Oct 23, 2023, at 11:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>>>>
>>>>>>
>>>>>>
>>>>>>> Am 23.10.2023 um 16:56 schrieb Qing Zhao <qing.zhao@oracle.com>:
>>>>>>>
>>>>>>>
>>>>>>>
>>>>>>>> On Oct 23, 2023, at 3:57 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>>>>>>>
>>>>>>>>> On Fri, Oct 20, 2023 at 10:41 PM Qing Zhao <qing.zhao@oracle.com> wrote:
>>>>>>>>>
>>>>>>>>>
>>>>>>>>>
>>>>>>>>>> On Oct 20, 2023, at 3:10 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>>>>>>>
>>>>>>>>>> On 2023-10-20 14:38, Qing Zhao wrote:
>>>>>>>>>>> How about the following:
>>>>>>>>>>> Add one more parameter to __builtin_dynamic_object_size(), i.e
>>>>>>>>>>> __builtin_dynamic_object_size (_1,1,array_annotated->foo)?
>>>>>>>>>>> When we see the structure field has counted_by attribute.
>>>>>>>>>>
>>>>>>>>>> Or maybe add a barrier preventing any assignments to array_annotated->foo from being reordered below the __bdos call? Basically an __asm__ with array_annotated->foo in the clobber list ought to do it I think.
>>>>>>>>>
>>>>>>>>> Maybe just adding the array_annotated->foo to the use list of the call to __builtin_dynamic_object_size should be enough?
>>>>>>>>>
>>>>>>>>> But I am not sure how to implement this in the TREE level, is there a USE_LIST/CLOBBER_LIST for each call? Then I can just simply add the counted_by field “array_annotated->foo” to the USE_LIST of the call to __bdos?
>>>>>>>>>
>>>>>>>>> This might be the simplest solution?
>>>>>>>>
>>>>>>>> If the dynamic object size is derived of a field then I think you need to
>>>>>>>> put the "load" of that memory location at the point (as argument)
>>>>>>>> of the __bos call right at parsing time. I know that's awkward because
>>>>>>>> you try to play tricks "discovering" that field only late, but that's not
>>>>>>>> going to work.
>>>>>>>
>>>>>>> Is it better to do this at gimplification phase instead of FE?
>>>>>>>
>>>>>>> VLA decls are handled in gimplification phase, the size calculation and call to alloca are all generated during this phase. (gimplify_vla_decl).
>>>>>>>
>>>>>>> For __bdos calls, we can add an additional argument if the object’s first argument’s type include the counted_by attribute, i.e
>>>>>>>
>>>>>>> ***During gimplification,
>>>>>>> For a call to __builtin_dynamic_object_size (ptr, type)
>>>>>>> Check whether the type of ptr includes counted_by attribute, if so, change the call to
>>>>>>> __builtin_dynamic_object_size (ptr, type, counted_by field)
>>>>>>>
>>>>>>> Then the correct data dependence should be represented well in the IR.
>>>>>>>
>>>>>>> **During object size phase,
>>>>>>>
>>>>>>> The call to __builtin_dynamic_object_size will become an expression includes the counted_by field or -1/0 when we cannot decide the size, the correct data dependence will be kept even the call to __builtin_dynamic_object_size is gone.
>>>>>>
>>>>>> But the whole point of the BOS pass is to derive information that is not available at parsing time, and that’s the cases you are after. The case where the connection to the field with the length is apparent during parsing is easy - you simply insert a load of the value before the BOS call.
>>>>>
>>>>> Yes, this is true.
>>>>> I prefer to implement this in gimplification phase since I am more familiar with the code there.. (I think that implementing it in gimplification should be very similar as implementing it in FE? Or do I miss anything here?)
>>>>>
>>>>> Joseph, if implement this in FE, where in the FE I should look at?
>>>>>
>>>>
>>>> We should aim for a good integration with the BDOS pass, so
>>>> that it can propagate the information further, e.g. the
>>>> following should work:
>>>>
>>>> struct { int L; char buf[] __counted_by(L) } x;
>>>> x.L = N;
>>>> x.buf = ...;
>>>> char *p = &x->f;
>>>> __bdos(p) -> N
>>>>
>>>> So we need to be smart on how we provide the size
>>>> information for x->f to the backend.
>>>
>>> To follow up on this. I do not think we should change the
>>> builtin in the FE or gimplification. Instead, we want
>>> to change the field access and compute the size there.
>> Could you please clarify on this? What do you mean by
>> "change the field access and compute the size there”?
>
> I think the FE should essentially give the
> type
>
> char [buf.L]
>
> to buf.x;
>
> If the type (or its size) could be preserved
> at this point so that it can be later
> discovered by __bdos, then it could know
> the size and propagate it further.
Currently, we already store the size info x.L of x.buf into the attribute list
of the field_decl of “x.buf”, __bdos readily to use it without any issue.
Putting “x.L” into TYPE of x.buf is the other approach, make it into a language extension.
So, Do you mean to implement the attribute similar as the language extension now?
i.e, convert the “attribute” info into the TYPE system at FE, then middle end will only use the TYPE info,
not the attribute anymore?
>
> For the attribute, this is not exactly what
> the FE could do because the semantic type
> can not change, but this is roughly the idea.
So, the attribute still cannot be put into the regular TYPE system at FE, we need to come up with new field in the current TYPE system to
Carry such info?
Then what’s the benefit from this new field in the TYPE system to my current approach (the attribute list of the field_decl)?
Can this new approach resolve the reordering issue?
>
>
>>>
>>> In my toy patch I then made this have a VLA type that
>>> encodes the size. Here, this would need to be done
>>> differently.
>>>
>>> But still, what we are missing in both cases
>>> is a proper way to pass the information down to BDOS.
>>
>> What’ s the issue with adding a new argument (x.L) to the BDOS call? What’s missing with this approach?
>>
>
> See the example above. the BDOS call might come much
> later when the relationship of the pointer to the
> field access is no longer there.
Why the relationship of the pointer to the field access is no longer there in _BDOS call in the above example?
My understanding is that the relationship still there, that is recorded in the attribute list of the
field_decl of the structure TYPE. BDOS call can access such information without any issue.
I tried to come up with a small testing case with your above example, but failed with a compilation error.
#include <stdint.h>
#include <malloc.h>
struct annotated {
size_t L;
char buf[] __attribute__((counted_by (L)));
};
int main ()
{
struct annotated x;
x.L = 10;
x.buf = (char *) malloc (x.L * sizeof (char));
char *p = &(x.buf);
size_t size = __builtin_dynamic_object_size (p, 1);
printf("the size of q is %lu \n", size);
return 0;
}
/home/opc/Install/latest-d/bin/gcc -O3 t4.c
t4.c: In function ‘main’:
t4.c:13:9: error: invalid use of flexible array member
13 | x.buf = (char *) malloc (x.L * sizeof (char));
| ^
t4.c:14:13: warning: initialization of ‘char *’ from incompatible pointer type ‘char (*)[]’ [-Wincompatible-pointer-types]
14 | char *p = &(x.buf);
| ^
Could you please provide me a working testing case for this?
On the other hand, the following small testing case works without any issue with my GCC:
#include <stdint.h>
#include <malloc.h>
struct annotated {
size_t foo;
char array[] __attribute__((counted_by (foo)));
};
#define noinline __attribute__((__noinline__))
static struct annotated * noinline alloc_buf (int index)
{
struct annotated *p;
p = malloc(sizeof (*p) + (index) * sizeof (char));
return p;
}
int main ()
{
size_t size = 0;
struct annotated *p = alloc_buf (10);
p->foo = 10;
char *q = p->array;
size = __builtin_dynamic_object_size (q, 1);
printf("the size of q is %lu \n", size);
return 0;
}
[opc@qinzhao-ol8u3-x86 Sid]$ sh t
/home/opc/Install/latest-d/bin/gcc -O3 t3.c
the size of q is 10
>
>>>
>>> For VLAs this works because BDOS can see the size of
>>> the definition. For calls to allocation functions
>>> it is read from an attribute.
>>
>> You mean for VLA, BDOS see the size of the definition
>> from the attribute for the allocation function?
>> Yes, that’s the case for VLA.
>
> Ok, I am wrong about how it works for VLAs. They
> get transformed to an alloca.
>
> But all calls marked with alloc_size and other
> allocations functions are detected in BDOS.
Yes.
Qing
>
>
>>
>> For VLA, the size computation and storage allocation are all done by the compiler (through “gimplify_vla_decl” in gimplification phase),
>> So these two can be tied together by the compiler.
>>
>> However, for FMA with counted_by attribute, the
>> storage allocation and the counted_by assignment
>> are done by the user.
>
> Yes.
>
> Martin
>
>>
>> Qing
>>>
>>> But I am not sure what would be the best way to encode
>>> this information so that BDOS can later access it.
>>>
>>> Martin
>>>
>>>
>>>
>>>
>>>>
>>>> This would also be desirable for the language extension.
>>>>
>>>> Martin
>>>>
>>>>
>>>>> Thanks a lot for the help.
>>>>>
>>>>> Qing
>>>>>
>>>>>> For the late case there’s no way to invent data flow dependence without inadvertently pessimizing optimization.
>>>>>>
>>>>>> Richard
>>>>>>
>>>>>>>
>>>>>>>>
>>>>>>>> A related issue is that assignment to the field and storage allocation
>>>>>>>> are not tied together
>>>>>>>
>>>>>>> Yes, this is different from VLA, in which, the size assignment and the storage allocation are generated and tied together by the compiler.
>>>>>>>
>>>>>>> For the flexible array member, the storage allocation and the size assignment are all done by the user. So, We need to clarify such requirement in the document to guide user to write correct code. And also, we might need to provide tools (warnings and sanitizer option) to help users to catch such coding error.
>>>>>>>
>>>>>>>> - if there's no use of the size data we might
>>>>>>>> remove the store of it as dead.
>>>>>>>
>>>>>>> Yes, when __bdos cannot decide the size, we need to remove the dead store to the field.
>>>>>>> I guess that the compiler should be able to do this automatically?
>>>>>>>
>>>>>>> thanks.
>>>>>>>
>>>>>>> Qing
>>>>>>>>
>>>>>>>> Of course I guess __bos then behaves like sizeof ().
>>>>>>>>
>>>>>>>> Richard.
>>>>>>>>
>>>>>>>>>
>>>>>>>>> Qing
>>>>>>>>>
>>>>>>>>>>
>>>>>>>>>> It may not work for something like this though:
>>>>>>>>>>
>>>>>>>>>> static size_t
>>>>>>>>>> get_size_of (void *ptr)
>>>>>>>>>> {
>>>>>>>>>> return __bdos (ptr, 1);
>>>>>>>>>> }
>>>>>>>>>>
>>>>>>>>>> void
>>>>>>>>>> foo (size_t sz)
>>>>>>>>>> {
>>>>>>>>>> array_annotated = __builtin_malloc (sz);
>>>>>>>>>> array_annotated = sz;
>>>>>>>>>>
>>>>>>>>>> ...
>>>>>>>>>> __builtin_printf ("%zu\n", get_size_of (array_annotated->foo));
>>>>>>>>>> ...
>>>>>>>>>> }
>>>>>>>>>>
>>>>>>>>>> because the call to get_size_of () may not have been inlined that early.
>>>>>>>>>>
>>>>>>>>>> The more fool-proof alternative may be to put a compile time barrier right below the assignment to array_annotated->foo; I reckon you could do that early in the front end by marking the size identifier and then tracking assignments to that identifier. That may have a slight runtime performance overhead since it may prevent even legitimate reordering. I can't think of another alternative at the moment...
>>>>>>>>>>
>>>>>>>>>> Sid
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 19:57 ` Martin Uecker
@ 2023-10-23 22:03 ` Kees Cook
0 siblings, 0 replies; 116+ messages in thread
From: Kees Cook @ 2023-10-23 22:03 UTC (permalink / raw)
To: Martin Uecker; +Cc: Qing Zhao, gcc-patches
On Mon, Oct 23, 2023 at 09:57:45PM +0200, Martin Uecker wrote:
> Am Montag, dem 23.10.2023 um 12:52 -0700 schrieb Kees Cook:
> > On Fri, Oct 20, 2023 at 09:54:05PM +0200, Martin Uecker wrote:
> > > Am Freitag, dem 20.10.2023 um 18:48 +0000 schrieb Qing Zhao:
> > > >
> > > > > On Oct 20, 2023, at 2:34 PM, Kees Cook <keescook@chromium.org> wrote:
> > > > >
> > > > > On Fri, Oct 20, 2023 at 11:50:11AM +0200, Martin Uecker wrote:
> > > > > > Am Donnerstag, dem 19.10.2023 um 16:33 -0700 schrieb Kees Cook:
> > > > > > > On Wed, Oct 18, 2023 at 09:11:43PM +0000, Qing Zhao wrote:
> > > > > > > > As I replied to Martin in another email, I plan to do the following to resolve this issue:
> > > > > > > >
> > > > > > > > 1. No specification for signed or unsigned for counted_by field.
> > > > > > > > 2. Add a sanitizer option -fsanitize=counted-by-bound to catch the cases when the size of the counted-by is not positive.
> > > > > > >
> > > > > > > I don't understand why this needs to be a runtime sanitizer. The
> > > > > > > signedness is known at compile time, so I would expect a -W option.
> > > > > >
> > > > > > The signedness of the type but not of the value.
> > > > > >
> > > > > > But I would not want to have a warning for signed
> > > > > > counter types by default because I would prefer
> > > > > > to use signed types (for various reasons including
> > > > > > better overflow detection).
> > > > > >
> > > > > > > Or
> > > > > > > do you mean you'd split up -fsanitize=bounds between unsigned and signed
> > > > > > > indexes? I'd find that kind of awkward for the kernel... but I feel like
> > > > > > > I've misunderstood something. :)
> > > > > > >
> > > > > > > -Kees
> > > > > >
> > > > > > The idea would be to detect at run-time the case
> > > > > > if x->buf is used at a time where x->counter
> > > > > > is negative and also when x->counter * sizeof(x->buf[0])
> > > > > > overflows or is too big.
> > > > > >
> > > > > > This would be similar to
> > > > > >
> > > > > > int a[n];
> > > > > >
> > > > > > where it is detected at run-time if n is not-positive.
> > > > >
> > > > > Right. I guess what I mean to say is that I would expect this case to
> > > > > already be caught by -fsanitize=bounds -- I don't see a reason to add an
> > > > > additional sanitizer option.
> > > > >
> > > > > struct foo {
> > > > > int count;
> > > > > int array[] __counted_by(count);
> > > > > };
> > > > >
> > > > > foo->count = 5;
> > > > > foo->array[0] = 1; // ok
> > > > > foo->array[10] = 1; // -fsanitize=bounds will catch this
> > > > > foo->array[-10] = 1; // -fsanitize=bounds will catch this too
> > > > >
> > > > >
> > > >
> > > > just checked this testing case with my GCC, and YES, -fsanitize=bounds indeed caught this error:
> > > >
> > > > ttt_1.c:31:12: runtime error: index 10 out of bounds for type 'char [*]'
> > > > ttt_1.c:32:12: runtime error: index -10 out of bounds for type 'char [*]’
> > > >
> > >
> > > Yes, but I thought we were discussing the case where count is
> > > set to a negative value:
> > >
> > > foo->count = -1;
> > > int x = foo->array[3]; // UBSan should diagnose this
> >
> > Oh right, I keep thinking about it backwards.
> >
> > Yeah, we can't trap the "count" assignment, because it may be getting used
> > for other purposes. But yeah, access to "array" should trap if "count"
> > is negative.
> >
> > > And also the case when foo->array becomes too big.
> >
> > How do you mean?
>
> count * sizeof(member) could overflow or otherwise be
> bigger than allowed.
Ah! Yes.
foo->count = SIZE_MAX;
foo->array[0]; // UBSan diagnose:
// SIZE_MAX * sizeof(int) is larger than can be represented
>
> Martin
>
>
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 19:43 ` Qing Zhao
@ 2023-10-23 22:48 ` Siddhesh Poyarekar
2023-10-24 20:30 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-23 22:48 UTC (permalink / raw)
To: Qing Zhao
Cc: Martin Uecker, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
On 2023-10-23 15:43, Qing Zhao wrote:
>
>
>> On Oct 23, 2023, at 2:43 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>
>> On 2023-10-23 14:06, Martin Uecker wrote:
>>> We should aim for a good integration with the BDOS pass, so
>>> that it can propagate the information further, e.g. the
>>> following should work:
>>> struct { int L; char buf[] __counted_by(L) } x;
>>> x.L = N;
>>> x.buf = ...;
>>> char *p = &x->f;
>>> __bdos(p) -> N
>>> So we need to be smart on how we provide the size
>>> information for x->f to the backend.
>>> This would also be desirable for the language extension.
>>
>> This is essentially why there need to be frontend rules constraining reordering and reachability semantics of x.L, thus restricting DSE and reordering for it.
>
> My understanding is that Restricting DSE and reordering should be done by the proper data flow information, with a new argument added to the BDOS call, this correct data flow information could be maintained, and then the DSE and reordering will not happen.
>
> I don’t quite understand what kind of frontend rules should be added to constrain reordering and reachability semantics? Can you explain this a little bit more? Do you mean to add some rules or requirment to the new attribute that the users of the attribute should follow in the source code?
Yes, but let me try and summarize the issues and the potential solutions
at the end:
>
>> This is not really a __bdos/__bos question, because that bit is trivial; if the structure is visible, the value is simply x.L. This is also why adding a reference to x.L in __bos/__bdos is not sufficient or even possible in, e.g. the above case you note.
>
> I am a little confused here, are we discussing how to resolve the potential reordering issue of the following:
>
> "
> struct annotated {
> size_t foo;
> char array[] __attribute__((counted_by (foo)));
> };
>
> p->foo = 10;
> size = __builtin_dynamic_object_size (p->array,1);
> “?
>
> Or a bigger issue?
Right, so the problem we're trying to solve is the reordering of __bdos
w.r.t. initialization of the size parameter but to also account for DSE
of the assignment, we can abstract this problem to that of DFA being
unable to see implicit use of the size parameter. __bdos is the one
such implicit user of the size parameter and you're proposing to solve
this by encoding the relationship between buffer and size at the __bdos
call site. But what about the case when the instantiation of the object
is not at the same place as the __bdos call site, i.e. the DFA is unable
to make that relationship?
The example Martin showed where the subobject gets "hidden" behind a
pointer was a trivial one where DFA *may* actually work in practice
(because the object-size pass can thread through these assignments) but
think about this one:
struct A
{
size_t size;
char buf[] __attribute__((counted_by(size)));
}
static size_t
get_size_of (void *ptr)
{
return __bdos (ptr, 1);
}
void
foo (size_t sz)
{
struct A *obj = __builtin_malloc (sz);
obj->size = sz;
...
__builtin_printf ("%zu\n", get_size_of (obj->array));
...
}
Until get_size_of is inlined, no DFA can see the __bdos call in the same
place as the point where obj is allocated. As a result, the assignment
to obj->size could get reordered (or the store eliminated) w.r.t. the
__bdos call until the inlining happens.
As a result, the relationship between buf and size established by the
attribute needs to be encoded into the type somehow. There are two options:
Option 1: Encode the relationship in the type of buf
This is kinda what you end up doing with component_ref_has_counted_by
and it does show the relationship if one is looking (through that call),
but nothing more that can be used to, e.g. prevent reordering or tell
the optimizer that the reference to the buf member may imply a reference
to the size member as well. This could be remedied by somehow encoding
the USES relationship for size into the type of buf that the
optimization passes can see. I feel like this may be a bit convoluted
to specify in a future language extension in a way that will actually be
well understood by developers, but it will likely generate faster
runtime code. This will also likely require a bigger change across passes.
Option 2: Encode the relationship in the type of size
The other option is to enhance the type of size somehow so that it
discourages reordering and store elimination, basically pessimizing
code. I think volatile semantics might be the way to do this and may
even be straightforward to specify in the future language extension
given that it builds on a known language construct and is thematically
related. However it does pessimize output for code that implements
__counted_by__.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-23 22:48 ` Siddhesh Poyarekar
@ 2023-10-24 20:30 ` Qing Zhao
2023-10-24 20:38 ` Martin Uecker
2023-10-24 21:03 ` Siddhesh Poyarekar
0 siblings, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-24 20:30 UTC (permalink / raw)
To: Siddhesh Poyarekar, Richard Biener
Cc: Martin Uecker, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
Hi, Sid,
Really appreciate for your example and detailed explanation. Very helpful.
I think that this example is an excellent example to show (almost) all the issues we need to consider.
I slightly modified this example to make it to be compilable and run-able, as following:
(but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
1 #include <malloc.h>
2 struct A
3 {
4 size_t size;
5 char buf[] __attribute__((counted_by(size)));
6 };
7
8 static size_t
9 get_size_from (void *ptr)
10 {
11 return __builtin_dynamic_object_size (ptr, 1);
12 }
13
14 void
15 foo (size_t sz)
16 {
17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
18 obj->size = sz;
19 obj->buf[0] = 2;
20 __builtin_printf (“%d\n", get_size_from (obj->buf));
21 return;
22 }
23
24 int main ()
25 {
26 foo (20);
27 return 0;
28 }
With my GCC, it was compiled and worked:
[opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
[opc@qinzhao-ol8u3-x86 ]$ ./a.out
20
Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
[opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
[opc@qinzhao-ol8u3-x86 ]$ ./a.out
-1
Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
Keep in mind of the above 2 situations, we will refer them in below:
1. First, the problem we are trying to resolve is:
(Your description):
> the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
basically is correct. However, with the following exception:
The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
2. Some facts I’d like to mention:
A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
2. Then, what’s the best approach to resolve this problem:
There were several suggestions so far:
A. Add an additional argument, the size parameter, to __bdos,
A.1, during FE;
A.2, during gimplification phase;
B. Encode the implicit USE in the type of size, to make the size “volatile”;
C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
As I explained in the above,
** Approach A (both A.1 and A.2) does not work;
** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
So, all the above proposed approaches, A, B, C, are not very good.
Then, maybe the following might work better?
In the tree optimization stage,
* After the inlining transformation applied,
+ * Before the data-flow related optimization happens,
+ * when the data flow analysis is constructed,
For each call to __bdos, add the implicit use of size parameter.
Is this doable?
Otherwise, we might need to take the “volatile” approach.
Let me know your suggestion and comment.
Thanks a lot.
Qing
> __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
>
> The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
>
> struct A
> {
> size_t size;
> char buf[] __attribute__((counted_by(size)));
> }
>
> static size_t
> get_size_of (void *ptr)
> {
> return __bdos (ptr, 1);
> }
>
> void
> foo (size_t sz)
> {
> struct A *obj = __builtin_malloc (sz);
> obj->size = sz;
>
> ...
> __builtin_printf ("%zu\n", get_size_of (obj->array));
> ...
> }
>
> Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
>
> As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
>
> Option 1: Encode the relationship in the type of buf
>
> This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
>
> Option 2: Encode the relationship in the type of size
>
> The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
>
> Thanks,
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 20:30 ` Qing Zhao
@ 2023-10-24 20:38 ` Martin Uecker
2023-10-24 21:09 ` Siddhesh Poyarekar
` (2 more replies)
2023-10-24 21:03 ` Siddhesh Poyarekar
1 sibling, 3 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-24 20:38 UTC (permalink / raw)
To: Qing Zhao, Siddhesh Poyarekar, Richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook, isanbard
Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> Hi, Sid,
>
> Really appreciate for your example and detailed explanation. Very helpful.
> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>
> I slightly modified this example to make it to be compilable and run-able, as following:
> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>
> 1 #include <malloc.h>
> 2 struct A
> 3 {
> 4 size_t size;
> 5 char buf[] __attribute__((counted_by(size)));
> 6 };
> 7
> 8 static size_t
> 9 get_size_from (void *ptr)
> 10 {
> 11 return __builtin_dynamic_object_size (ptr, 1);
> 12 }
> 13
> 14 void
> 15 foo (size_t sz)
> 16 {
> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> 18 obj->size = sz;
> 19 obj->buf[0] = 2;
> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> 21 return;
> 22 }
> 23
> 24 int main ()
> 25 {
> 26 foo (20);
> 27 return 0;
> 28 }
>
> With my GCC, it was compiled and worked:
> [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> 20
> Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
>
> [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> -1
> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>
> Keep in mind of the above 2 situations, we will refer them in below:
>
> 1. First, the problem we are trying to resolve is:
>
> (Your description):
>
> > the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
>
> basically is correct. However, with the following exception:
>
> The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
> In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
>
> With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
>
> As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
>
> 2. Some facts I’d like to mention:
>
> A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
>
> B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
>
> 2. Then, what’s the best approach to resolve this problem:
>
> There were several suggestions so far:
>
> A. Add an additional argument, the size parameter, to __bdos,
> A.1, during FE;
> A.2, during gimplification phase;
> B. Encode the implicit USE in the type of size, to make the size “volatile”;
> C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
>
> As I explained in the above,
> ** Approach A (both A.1 and A.2) does not work;
> ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
> ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
>
> So, all the above proposed approaches, A, B, C, are not very good.
>
> Then, maybe the following might work better?
>
> In the tree optimization stage,
> * After the inlining transformation applied,
> + * Before the data-flow related optimization happens,
> + * when the data flow analysis is constructed,
>
> For each call to __bdos, add the implicit use of size parameter.
>
> Is this doable?
Here is another proposal: Add a new builtin function
__builtin_with_size(x, size)
that return x but behaves similar to an allocation
function in that BDOS can look at the size argument
to discover the size.
The FE insers this function when the field is accessed:
__builtin_with_size(x.buf, x.L);
Martin
>
> Otherwise, we might need to take the “volatile” approach.
>
> Let me know your suggestion and comment.
>
> Thanks a lot.
>
> Qing
>
>
> > __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
> >
> > The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
> >
> > struct A
> > {
> > size_t size;
> > char buf[] __attribute__((counted_by(size)));
> > }
> >
> > static size_t
> > get_size_of (void *ptr)
> > {
> > return __bdos (ptr, 1);
> > }
> >
> > void
> > foo (size_t sz)
> > {
> > struct A *obj = __builtin_malloc (sz);
> > obj->size = sz;
> >
> > ...
> > __builtin_printf ("%zu\n", get_size_of (obj->array));
> > ...
> > }
> >
> > Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
> >
> > As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
> >
> > Option 1: Encode the relationship in the type of buf
> >
> > This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
> >
> > Option 2: Encode the relationship in the type of size
> >
> > The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
> >
> > Thanks,
> > Sid
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 20:30 ` Qing Zhao
2023-10-24 20:38 ` Martin Uecker
@ 2023-10-24 21:03 ` Siddhesh Poyarekar
2023-10-24 22:41 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-24 21:03 UTC (permalink / raw)
To: Qing Zhao, Richard Biener
Cc: Martin Uecker, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
On 2023-10-24 16:30, Qing Zhao wrote:
> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>
But __bos/__bdos are barely useful without optimization; you need a
minimum of -O1. You're right that if the call is never inlined then we
don't care because the __bdos call does not get expanded to obj->size.
However, the point of situation 2 is that the TYPE info cannot be used
by the __bdos call *only for a while* (i.e. until the call gets inlined)
and that window is an opportunity for the reordering/DSE to break things.
Thanks.
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 20:38 ` Martin Uecker
@ 2023-10-24 21:09 ` Siddhesh Poyarekar
2023-10-24 22:51 ` Qing Zhao
2023-10-25 6:43 ` Richard Biener
2 siblings, 0 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-24 21:09 UTC (permalink / raw)
To: Martin Uecker, Qing Zhao, Richard Biener
Cc: Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook, isanbard
On 2023-10-24 16:38, Martin Uecker wrote:
> Here is another proposal: Add a new builtin function
>
> __builtin_with_size(x, size)
>
> that return x but behaves similar to an allocation
> function in that BDOS can look at the size argument
> to discover the size.
>
> The FE insers this function when the field is accessed:
>
> __builtin_with_size(x.buf, x.L);
>
In fact if we do this at the allocation site for x, it may also help
with future warnings, where the compiler could flag a warning or error
when it encounters this builtin but does not see an assignment to x.L.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 21:03 ` Siddhesh Poyarekar
@ 2023-10-24 22:41 ` Qing Zhao
2023-10-24 23:51 ` Siddhesh Poyarekar
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-24 22:41 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Richard Biener, Martin Uecker, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 24, 2023, at 5:03 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-24 16:30, Qing Zhao wrote:
>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>
> But __bos/__bdos are barely useful without optimization; you need a minimum of -O1. You're right that if the call is never inlined then we don't care because the __bdos call does not get expanded to obj->size.
>
> However, the point of situation 2 is that the TYPE info cannot be used by the __bdos call *only for a while* (i.e. until the call gets inlined) and that window is an opportunity for the reordering/DSE to break things.
The main point of situation 2 I tried made: there are situations where obj->size is not used at all by the __bdos, marking it as volatile is too conservative, unnecessarily prevent useful optimizations from happening. -:)
Qing
>
> Thanks.
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 20:38 ` Martin Uecker
2023-10-24 21:09 ` Siddhesh Poyarekar
@ 2023-10-24 22:51 ` Qing Zhao
2023-10-24 23:56 ` Siddhesh Poyarekar
2023-10-25 5:26 ` Martin Uecker
2023-10-25 6:43 ` Richard Biener
2 siblings, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-24 22:51 UTC (permalink / raw)
To: Martin Uecker
Cc: Siddhesh Poyarekar, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 24, 2023, at 4:38 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>> Hi, Sid,
>>
>> Really appreciate for your example and detailed explanation. Very helpful.
>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>
>> I slightly modified this example to make it to be compilable and run-able, as following:
>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>
>> 1 #include <malloc.h>
>> 2 struct A
>> 3 {
>> 4 size_t size;
>> 5 char buf[] __attribute__((counted_by(size)));
>> 6 };
>> 7
>> 8 static size_t
>> 9 get_size_from (void *ptr)
>> 10 {
>> 11 return __builtin_dynamic_object_size (ptr, 1);
>> 12 }
>> 13
>> 14 void
>> 15 foo (size_t sz)
>> 16 {
>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>> 18 obj->size = sz;
>> 19 obj->buf[0] = 2;
>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>> 21 return;
>> 22 }
>> 23
>> 24 int main ()
>> 25 {
>> 26 foo (20);
>> 27 return 0;
>> 28 }
>>
>> With my GCC, it was compiled and worked:
>> [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>> 20
>> Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
>>
>> [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>> -1
>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>>
>> Keep in mind of the above 2 situations, we will refer them in below:
>>
>> 1. First, the problem we are trying to resolve is:
>>
>> (Your description):
>>
>>> the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
>>
>> basically is correct. However, with the following exception:
>>
>> The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
>> In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
>>
>> With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
>>
>> As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
>>
>> 2. Some facts I’d like to mention:
>>
>> A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
>>
>> B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
>>
>> 2. Then, what’s the best approach to resolve this problem:
>>
>> There were several suggestions so far:
>>
>> A. Add an additional argument, the size parameter, to __bdos,
>> A.1, during FE;
>> A.2, during gimplification phase;
>> B. Encode the implicit USE in the type of size, to make the size “volatile”;
>> C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
>>
>> As I explained in the above,
>> ** Approach A (both A.1 and A.2) does not work;
>> ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
>> ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
>>
>> So, all the above proposed approaches, A, B, C, are not very good.
>>
>> Then, maybe the following might work better?
>>
>> In the tree optimization stage,
>> * After the inlining transformation applied,
>> + * Before the data-flow related optimization happens,
>> + * when the data flow analysis is constructed,
>>
>> For each call to __bdos, add the implicit use of size parameter.
>>
>> Is this doable?
>
> Here is another proposal: Add a new builtin function
>
> __builtin_with_size(x, size)
>
> that return x but behaves similar to an allocation
> function in that BDOS can look at the size argument
> to discover the size.
>
> The FE insers this function when the field is accessed:
>
> __builtin_with_size(x.buf, x.L);
Thanks for the proposal!
So what you suggested is:
For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
Become:
_bdos(__builtin_with_size(x.buf, x.L), 1)?
Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
This looks like a very promising solution.
Will study this a. Little bit more.
Qing
>
>
> Martin
>
>
>
>>
>> Otherwise, we might need to take the “volatile” approach.
>>
>> Let me know your suggestion and comment.
>>
>> Thanks a lot.
>>
>> Qing
>>
>>
>>> __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
>>>
>>> The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
>>>
>>> struct A
>>> {
>>> size_t size;
>>> char buf[] __attribute__((counted_by(size)));
>>> }
>>>
>>> static size_t
>>> get_size_of (void *ptr)
>>> {
>>> return __bdos (ptr, 1);
>>> }
>>>
>>> void
>>> foo (size_t sz)
>>> {
>>> struct A *obj = __builtin_malloc (sz);
>>> obj->size = sz;
>>>
>>> ...
>>> __builtin_printf ("%zu\n", get_size_of (obj->array));
>>> ...
>>> }
>>>
>>> Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
>>>
>>> As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
>>>
>>> Option 1: Encode the relationship in the type of buf
>>>
>>> This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
>>>
>>> Option 2: Encode the relationship in the type of size
>>>
>>> The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
>>>
>>> Thanks,
>>> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 22:41 ` Qing Zhao
@ 2023-10-24 23:51 ` Siddhesh Poyarekar
2023-10-25 21:59 ` Kees Cook
0 siblings, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-24 23:51 UTC (permalink / raw)
To: Qing Zhao
Cc: Richard Biener, Martin Uecker, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
On 2023-10-24 18:41, Qing Zhao wrote:
>
>
>> On Oct 24, 2023, at 5:03 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>
>> On 2023-10-24 16:30, Qing Zhao wrote:
>>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>>
>> But __bos/__bdos are barely useful without optimization; you need a minimum of -O1. You're right that if the call is never inlined then we don't care because the __bdos call does not get expanded to obj->size.
>>
>> However, the point of situation 2 is that the TYPE info cannot be used by the __bdos call *only for a while* (i.e. until the call gets inlined) and that window is an opportunity for the reordering/DSE to break things.
>
> The main point of situation 2 I tried made: there are situations where obj->size is not used at all by the __bdos, marking it as volatile is too conservative, unnecessarily prevent useful optimizations from happening. -:)
Yes, that's the tradeoff. However, maybe this is the point where Kees
jumps in and say the kernel doesn't really care as much or something
like that :)
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 22:51 ` Qing Zhao
@ 2023-10-24 23:56 ` Siddhesh Poyarekar
2023-10-25 13:27 ` Qing Zhao
2023-10-25 5:26 ` Martin Uecker
1 sibling, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-24 23:56 UTC (permalink / raw)
To: Qing Zhao, Martin Uecker
Cc: Richard Biener, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
On 2023-10-24 18:51, Qing Zhao wrote:
> Thanks for the proposal!
>
> So what you suggested is:
>
> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
> Become:
>
> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>
> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
Oops, I think Martin and I fell off-list in a subthread. I clarified
that my comment was that any such annotation at object reference is
probably too late and hence not the right place for it; basically it has
the same problems as the option A in your comment. A better place to
reinforce such a relationship would be the allocation+initialization
site instead.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 22:51 ` Qing Zhao
2023-10-24 23:56 ` Siddhesh Poyarekar
@ 2023-10-25 5:26 ` Martin Uecker
1 sibling, 0 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-25 5:26 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Dienstag, dem 24.10.2023 um 22:51 +0000 schrieb Qing Zhao:
>
> > On Oct 24, 2023, at 4:38 PM, Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > Hi, Sid,
> > >
> > > Really appreciate for your example and detailed explanation. Very helpful.
> > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > >
> > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > >
> > > 1 #include <malloc.h>
> > > 2 struct A
> > > 3 {
> > > 4 size_t size;
> > > 5 char buf[] __attribute__((counted_by(size)));
> > > 6 };
> > > 7
> > > 8 static size_t
> > > 9 get_size_from (void *ptr)
> > > 10 {
> > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > 12 }
> > > 13
> > > 14 void
> > > 15 foo (size_t sz)
> > > 16 {
> > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > 18 obj->size = sz;
> > > 19 obj->buf[0] = 2;
> > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > 21 return;
> > > 22 }
> > > 23
> > > 24 int main ()
> > > 25 {
> > > 26 foo (20);
> > > 27 return 0;
> > > 28 }
> > >
> > > With my GCC, it was compiled and worked:
> > > [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
> > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > 20
> > > Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
> > >
> > > [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
> > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > -1
> > > Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
> > >
> > > Keep in mind of the above 2 situations, we will refer them in below:
> > >
> > > 1. First, the problem we are trying to resolve is:
> > >
> > > (Your description):
> > >
> > > > the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
> > >
> > > basically is correct. However, with the following exception:
> > >
> > > The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
> > > In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
> > >
> > > With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
> > >
> > > As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
> > >
> > > 2. Some facts I’d like to mention:
> > >
> > > A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
> > >
> > > B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
> > >
> > > 2. Then, what’s the best approach to resolve this problem:
> > >
> > > There were several suggestions so far:
> > >
> > > A. Add an additional argument, the size parameter, to __bdos,
> > > A.1, during FE;
> > > A.2, during gimplification phase;
> > > B. Encode the implicit USE in the type of size, to make the size “volatile”;
> > > C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
> > >
> > > As I explained in the above,
> > > ** Approach A (both A.1 and A.2) does not work;
> > > ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
> > > ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
> > >
> > > So, all the above proposed approaches, A, B, C, are not very good.
> > >
> > > Then, maybe the following might work better?
> > >
> > > In the tree optimization stage,
> > > * After the inlining transformation applied,
> > > + * Before the data-flow related optimization happens,
> > > + * when the data flow analysis is constructed,
> > >
> > > For each call to __bdos, add the implicit use of size parameter.
> > >
> > > Is this doable?
> >
> > Here is another proposal: Add a new builtin function
> >
> > __builtin_with_size(x, size)
> >
> > that return x but behaves similar to an allocation
> > function in that BDOS can look at the size argument
> > to discover the size.
> >
> > The FE insers this function when the field is accessed:
> >
> > __builtin_with_size(x.buf, x.L);
>
> Thanks for the proposal!
>
> So what you suggested is:
>
> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
> Become:
>
> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>
> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>
> This looks like a very promising solution.
>
> Will study this a. Little bit more.
Yes, the load will be created explicitely in the FE
at the right position. The BDOS pass can then
later propagate the size to where it is used:
x = &__builtin_with_size(x.buf, x.L)
...other stuff is happening...
__bdos(x, 1).
See for a working example: https://godbolt.org/z/Ej3s1GToa
which shows that reordering is still possible.
I think this should be easy to implement
because it is similar to how BDOS works with
builtin allocation functions.
And this feature seems generally useful.
I am not sure whether the builtin to take a pointer
argument or the object itself.
Note that the builtin does nothing. It just ensure
that the size argument is evaluated at the right
point in time.
Maybe it should have arguments for min / max
subobject etc.. Not sure.
Martin
>
> Qing
> >
> >
> > Martin
> >
> >
> >
> > >
> > > Otherwise, we might need to take the “volatile” approach.
> > >
> > > Let me know your suggestion and comment.
> > >
> > > Thanks a lot.
> > >
> > > Qing
> > >
> > >
> > > > __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
> > > >
> > > > The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
> > > >
> > > > struct A
> > > > {
> > > > size_t size;
> > > > char buf[] __attribute__((counted_by(size)));
> > > > }
> > > >
> > > > static size_t
> > > > get_size_of (void *ptr)
> > > > {
> > > > return __bdos (ptr, 1);
> > > > }
> > > >
> > > > void
> > > > foo (size_t sz)
> > > > {
> > > > struct A *obj = __builtin_malloc (sz);
> > > > obj->size = sz;
> > > >
> > > > ...
> > > > __builtin_printf ("%zu\n", get_size_of (obj->array));
> > > > ...
> > > > }
> > > >
> > > > Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
> > > >
> > > > As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
> > > >
> > > > Option 1: Encode the relationship in the type of buf
> > > >
> > > > This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
> > > >
> > > > Option 2: Encode the relationship in the type of size
> > > >
> > > > The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
> > > >
> > > > Thanks,
> > > > Sid
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 20:38 ` Martin Uecker
2023-10-24 21:09 ` Siddhesh Poyarekar
2023-10-24 22:51 ` Qing Zhao
@ 2023-10-25 6:43 ` Richard Biener
2023-10-25 8:16 ` Martin Uecker
2 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-25 6:43 UTC (permalink / raw)
To: Martin Uecker
Cc: Qing Zhao, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>
> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>> Hi, Sid,
>>
>> Really appreciate for your example and detailed explanation. Very helpful.
>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>
>> I slightly modified this example to make it to be compilable and run-able, as following:
>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>
>> 1 #include <malloc.h>
>> 2 struct A
>> 3 {
>> 4 size_t size;
>> 5 char buf[] __attribute__((counted_by(size)));
>> 6 };
>> 7
>> 8 static size_t
>> 9 get_size_from (void *ptr)
>> 10 {
>> 11 return __builtin_dynamic_object_size (ptr, 1);
>> 12 }
>> 13
>> 14 void
>> 15 foo (size_t sz)
>> 16 {
>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>> 18 obj->size = sz;
>> 19 obj->buf[0] = 2;
>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>> 21 return;
>> 22 }
>> 23
>> 24 int main ()
>> 25 {
>> 26 foo (20);
>> 27 return 0;
>> 28 }
>>
>> With my GCC, it was compiled and worked:
>> [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>> 20
>> Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
>>
>> [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>> -1
>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>>
>> Keep in mind of the above 2 situations, we will refer them in below:
>>
>> 1. First, the problem we are trying to resolve is:
>>
>> (Your description):
>>
>>> the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
>>
>> basically is correct. However, with the following exception:
>>
>> The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
>> In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
>>
>> With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
>>
>> As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
>>
>> 2. Some facts I’d like to mention:
>>
>> A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
>>
>> B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
>>
>> 2. Then, what’s the best approach to resolve this problem:
>>
>> There were several suggestions so far:
>>
>> A. Add an additional argument, the size parameter, to __bdos,
>> A.1, during FE;
>> A.2, during gimplification phase;
>> B. Encode the implicit USE in the type of size, to make the size “volatile”;
>> C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
>>
>> As I explained in the above,
>> ** Approach A (both A.1 and A.2) does not work;
>> ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
>> ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
>>
>> So, all the above proposed approaches, A, B, C, are not very good.
>>
>> Then, maybe the following might work better?
>>
>> In the tree optimization stage,
>> * After the inlining transformation applied,
>> + * Before the data-flow related optimization happens,
>> + * when the data flow analysis is constructed,
>>
>> For each call to __bdos, add the implicit use of size parameter.
>>
>> Is this doable?
>
> Here is another proposal: Add a new builtin function
>
> __builtin_with_size(x, size)
>
> that return x but behaves similar to an allocation
> function in that BDOS can look at the size argument
> to discover the size.
>
> The FE insers this function when the field is accessed:
When it’s set I suppose. Turn
X.l = n;
Into
X.l = __builtin_with_size (x.buf, n);
And indeed we need sth like a fat pointer to reliably solve all the issues.
Richard
> __builtin_with_size(x.buf, x.L);
>
>
> Martin
>
>
>
>>
>> Otherwise, we might need to take the “volatile” approach.
>>
>> Let me know your suggestion and comment.
>>
>> Thanks a lot.
>>
>> Qing
>>
>>
>>> __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
>>>
>>> The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
>>>
>>> struct A
>>> {
>>> size_t size;
>>> char buf[] __attribute__((counted_by(size)));
>>> }
>>>
>>> static size_t
>>> get_size_of (void *ptr)
>>> {
>>> return __bdos (ptr, 1);
>>> }
>>>
>>> void
>>> foo (size_t sz)
>>> {
>>> struct A *obj = __builtin_malloc (sz);
>>> obj->size = sz;
>>>
>>> ...
>>> __builtin_printf ("%zu\n", get_size_of (obj->array));
>>> ...
>>> }
>>>
>>> Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
>>>
>>> As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
>>>
>>> Option 1: Encode the relationship in the type of buf
>>>
>>> This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
>>>
>>> Option 2: Encode the relationship in the type of size
>>>
>>> The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
>>>
>>> Thanks,
>>> Sid
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 6:43 ` Richard Biener
@ 2023-10-25 8:16 ` Martin Uecker
2023-10-25 10:25 ` Siddhesh Poyarekar
2023-10-25 10:25 ` Richard Biener
0 siblings, 2 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-25 8:16 UTC (permalink / raw)
To: Richard Biener
Cc: Qing Zhao, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>
> > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> >
> > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > Hi, Sid,
> > >
> > > Really appreciate for your example and detailed explanation. Very helpful.
> > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > >
> > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > >
> > > 1 #include <malloc.h>
> > > 2 struct A
> > > 3 {
> > > 4 size_t size;
> > > 5 char buf[] __attribute__((counted_by(size)));
> > > 6 };
> > > 7
> > > 8 static size_t
> > > 9 get_size_from (void *ptr)
> > > 10 {
> > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > 12 }
> > > 13
> > > 14 void
> > > 15 foo (size_t sz)
> > > 16 {
> > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > 18 obj->size = sz;
> > > 19 obj->buf[0] = 2;
> > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > 21 return;
> > > 22 }
> > > 23
> > > 24 int main ()
> > > 25 {
> > > 26 foo (20);
> > > 27 return 0;
> > > 28 }
> > >
> > > With my GCC, it was compiled and worked:
> > > [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
> > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > 20
> > > Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
> > >
> > > [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
> > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > -1
> > > Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
> > >
> > > Keep in mind of the above 2 situations, we will refer them in below:
> > >
> > > 1. First, the problem we are trying to resolve is:
> > >
> > > (Your description):
> > >
> > > > the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
> > >
> > > basically is correct. However, with the following exception:
> > >
> > > The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
> > > In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
> > >
> > > With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
> > >
> > > As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
> > >
> > > 2. Some facts I’d like to mention:
> > >
> > > A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
> > >
> > > B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
> > >
> > > 2. Then, what’s the best approach to resolve this problem:
> > >
> > > There were several suggestions so far:
> > >
> > > A. Add an additional argument, the size parameter, to __bdos,
> > > A.1, during FE;
> > > A.2, during gimplification phase;
> > > B. Encode the implicit USE in the type of size, to make the size “volatile”;
> > > C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
> > >
> > > As I explained in the above,
> > > ** Approach A (both A.1 and A.2) does not work;
> > > ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
> > > ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
> > >
> > > So, all the above proposed approaches, A, B, C, are not very good.
> > >
> > > Then, maybe the following might work better?
> > >
> > > In the tree optimization stage,
> > > * After the inlining transformation applied,
> > > + * Before the data-flow related optimization happens,
> > > + * when the data flow analysis is constructed,
> > >
> > > For each call to __bdos, add the implicit use of size parameter.
> > >
> > > Is this doable?
> >
> > Here is another proposal: Add a new builtin function
> >
> > __builtin_with_size(x, size)
> >
> > that return x but behaves similar to an allocation
> > function in that BDOS can look at the size argument
> > to discover the size.
> >
> > The FE insers this function when the field is accessed:
>
> When it’s set I suppose. Turn
>
> X.l = n;
>
> Into
>
> X.l = __builtin_with_size (x.buf, n);
It would turn
some_variable = (&) x.buf
into
some_variable = __builtin_with_size ( (&) x.buf. x.len)
So the later access to x.buf and not the initialization
of a member of the struct (which is too early).
>
> And indeed we need sth like a fat pointer to reliably solve all the issues.
What happens for other languages such as FORTRAN
and ADA do? Are those pointers lowered in the FE?
To me it seems there are two sound ways to introduce
such information:
- either by using the type system. This works in
the FE in C using variably modified types
char buf[n];
__auto_type p = &buf;
... = sizeof (*p);
But if I understand Jakob's comment to some PR
correctly the size information in the TREE_TYPE
is not processed correctly anymore in the
middle-end.
- or one injects the information via some
tree node or builtin at certain points in
time as suggested here, and the compiler
derives the information from these points
as tree-object-size does.
The use of attributes seems fragile and - looking
at the access attribute also overly complex. And
we somehow support this only for function types
and not elsewhere and also this then gets lost
during inlining. So I think for all this stuff
(nonnull, access, counted_by) I think a better
approach is needed.
Martin
>
> Richard
>
> > __builtin_with_size(x.buf, x.L);
> >
> >
> > Martin
> >
> >
> >
> > >
> > > Otherwise, we might need to take the “volatile” approach.
> > >
> > > Let me know your suggestion and comment.
> > >
> > > Thanks a lot.
> > >
> > > Qing
> > >
> > >
> > > > __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
> > > >
> > > > The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
> > > >
> > > > struct A
> > > > {
> > > > size_t size;
> > > > char buf[] __attribute__((counted_by(size)));
> > > > }
> > > >
> > > > static size_t
> > > > get_size_of (void *ptr)
> > > > {
> > > > return __bdos (ptr, 1);
> > > > }
> > > >
> > > > void
> > > > foo (size_t sz)
> > > > {
> > > > struct A *obj = __builtin_malloc (sz);
> > > > obj->size = sz;
> > > >
> > > > ...
> > > > __builtin_printf ("%zu\n", get_size_of (obj->array));
> > > > ...
> > > > }
> > > >
> > > > Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
> > > >
> > > > As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
> > > >
> > > > Option 1: Encode the relationship in the type of buf
> > > >
> > > > This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
> > > >
> > > > Option 2: Encode the relationship in the type of size
> > > >
> > > > The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
> > > >
> > > > Thanks,
> > > > Sid
> > >
> >
--
Univ.-Prof. Dr. rer. nat. Martin Uecker
Graz University of Technology
Institute of Biomedical Imaging
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 8:16 ` Martin Uecker
@ 2023-10-25 10:25 ` Siddhesh Poyarekar
2023-10-25 10:47 ` Martin Uecker
2023-10-25 10:25 ` Richard Biener
1 sibling, 1 reply; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-25 10:25 UTC (permalink / raw)
To: Martin Uecker, Richard Biener
Cc: Qing Zhao, Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook, isanbard
On 2023-10-25 04:16, Martin Uecker wrote:
> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>
>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>
>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>> Hi, Sid,
>>>>
>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>
>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>
>>>> 1 #include <malloc.h>
>>>> 2 struct A
>>>> 3 {
>>>> 4 size_t size;
>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>> 6 };
>>>> 7
>>>> 8 static size_t
>>>> 9 get_size_from (void *ptr)
>>>> 10 {
>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>> 12 }
>>>> 13
>>>> 14 void
>>>> 15 foo (size_t sz)
>>>> 16 {
>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>> 18 obj->size = sz;
>>>> 19 obj->buf[0] = 2;
>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>> 21 return;
>>>> 22 }
>>>> 23
>>>> 24 int main ()
>>>> 25 {
>>>> 26 foo (20);
>>>> 27 return 0;
>>>> 28 }
>>>>
<snip>
>> When it’s set I suppose. Turn
>>
>> X.l = n;
>>
>> Into
>>
>> X.l = __builtin_with_size (x.buf, n);
>
> It would turn
>
> some_variable = (&) x.buf
>
> into
>
> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>
>
> So the later access to x.buf and not the initialization
> of a member of the struct (which is too early).
>
Hmm, so with Qing's example above, are you suggesting the transformation
be to foo like so:
14 void
15 foo (size_t sz)
16 {
16.5 void * _1;
17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
18 obj->size = sz;
19 obj->buf[0] = 2;
19.5 _1 = __builtin_with_size (obj->buf, obj->size);
20 __builtin_printf (“%d\n", get_size_from (_1));
21 return;
22 }
If yes then this could indeed work. I think I got thrown off by the
reference to __bdos.
Thanks,
Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 8:16 ` Martin Uecker
2023-10-25 10:25 ` Siddhesh Poyarekar
@ 2023-10-25 10:25 ` Richard Biener
2023-10-25 10:39 ` Martin Uecker
1 sibling, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-25 10:25 UTC (permalink / raw)
To: Martin Uecker
Cc: Qing Zhao, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 25.10.2023 um 10:16 schrieb Martin Uecker <uecker@tugraz.at>:
>
> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>
>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>
>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>> Hi, Sid,
>>>>
>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>
>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>
>>>> 1 #include <malloc.h>
>>>> 2 struct A
>>>> 3 {
>>>> 4 size_t size;
>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>> 6 };
>>>> 7
>>>> 8 static size_t
>>>> 9 get_size_from (void *ptr)
>>>> 10 {
>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>> 12 }
>>>> 13
>>>> 14 void
>>>> 15 foo (size_t sz)
>>>> 16 {
>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>> 18 obj->size = sz;
>>>> 19 obj->buf[0] = 2;
>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>> 21 return;
>>>> 22 }
>>>> 23
>>>> 24 int main ()
>>>> 25 {
>>>> 26 foo (20);
>>>> 27 return 0;
>>>> 28 }
>>>>
>>>> With my GCC, it was compiled and worked:
>>>> [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
>>>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>>>> 20
>>>> Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
>>>>
>>>> [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
>>>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>>>> -1
>>>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>>>>
>>>> Keep in mind of the above 2 situations, we will refer them in below:
>>>>
>>>> 1. First, the problem we are trying to resolve is:
>>>>
>>>> (Your description):
>>>>
>>>>> the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
>>>>
>>>> basically is correct. However, with the following exception:
>>>>
>>>> The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
>>>> In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
>>>>
>>>> With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
>>>>
>>>> As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
>>>>
>>>> 2. Some facts I’d like to mention:
>>>>
>>>> A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
>>>>
>>>> B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
>>>>
>>>> 2. Then, what’s the best approach to resolve this problem:
>>>>
>>>> There were several suggestions so far:
>>>>
>>>> A. Add an additional argument, the size parameter, to __bdos,
>>>> A.1, during FE;
>>>> A.2, during gimplification phase;
>>>> B. Encode the implicit USE in the type of size, to make the size “volatile”;
>>>> C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
>>>>
>>>> As I explained in the above,
>>>> ** Approach A (both A.1 and A.2) does not work;
>>>> ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
>>>> ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
>>>>
>>>> So, all the above proposed approaches, A, B, C, are not very good.
>>>>
>>>> Then, maybe the following might work better?
>>>>
>>>> In the tree optimization stage,
>>>> * After the inlining transformation applied,
>>>> + * Before the data-flow related optimization happens,
>>>> + * when the data flow analysis is constructed,
>>>>
>>>> For each call to __bdos, add the implicit use of size parameter.
>>>>
>>>> Is this doable?
>>>
>>> Here is another proposal: Add a new builtin function
>>>
>>> __builtin_with_size(x, size)
>>>
>>> that return x but behaves similar to an allocation
>>> function in that BDOS can look at the size argument
>>> to discover the size.
>>>
>>> The FE insers this function when the field is accessed:
>>
>> When it’s set I suppose. Turn
>>
>> X.l = n;
>>
>> Into
>>
>> X.l = __builtin_with_size (x.buf, n);
>
> It would turn
>
> some_variable = (&) x.buf
>
> into
>
> some_variable = __builtin_with_size ( (&) x.buf. x.len)
Unless you use the address of x.Len this will not work when len is initialized after buf. And the address will not have a meaningful data dependence.
>
> So the later access to x.buf and not the initialization
> of a member of the struct (which is too early).
>>
>> And indeed we need sth like a fat pointer to reliably solve all the issues.
>
> What happens for other languages such as FORTRAN
> and ADA do? Are those pointers lowered in the FE?
Yes
> To me it seems there are two sound ways to introduce
> such information:
>
> - either by using the type system. This works in
> the FE in C using variably modified types
>
> char buf[n];
> __auto_type p = &buf;
>
> ... = sizeof (*p);
>
> But if I understand Jakob's comment to some PR
> correctly the size information in the TREE_TYPE
> is not processed correctly anymore in the
> middle-end.
The type based info is lowered during gimplification and in particular for pointer types the middle-end quickly loses track of the original type.
Richard
>
> - or one injects the information via some
> tree node or builtin at certain points in
> time as suggested here, and the compiler
> derives the information from these points
> as tree-object-size does.
>
>
> The use of attributes seems fragile and - looking
> at the access attribute also overly complex. And
> we somehow support this only for function types
> and not elsewhere and also this then gets lost
> during inlining. So I think for all this stuff
> (nonnull, access, counted_by) I think a better
> approach is needed.
>
>
> Martin
>
>
>>
>> Richard
>
>
>
>
>>
>>> __builtin_with_size(x.buf, x.L);
>>>
>>>
>>> Martin
>>>
>>>
>>>
>>>>
>>>> Otherwise, we might need to take the “volatile” approach.
>>>>
>>>> Let me know your suggestion and comment.
>>>>
>>>> Thanks a lot.
>>>>
>>>> Qing
>>>>
>>>>
>>>>> __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
>>>>>
>>>>> The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
>>>>>
>>>>> struct A
>>>>> {
>>>>> size_t size;
>>>>> char buf[] __attribute__((counted_by(size)));
>>>>> }
>>>>>
>>>>> static size_t
>>>>> get_size_of (void *ptr)
>>>>> {
>>>>> return __bdos (ptr, 1);
>>>>> }
>>>>>
>>>>> void
>>>>> foo (size_t sz)
>>>>> {
>>>>> struct A *obj = __builtin_malloc (sz);
>>>>> obj->size = sz;
>>>>>
>>>>> ...
>>>>> __builtin_printf ("%zu\n", get_size_of (obj->array));
>>>>> ...
>>>>> }
>>>>>
>>>>> Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
>>>>>
>>>>> As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
>>>>>
>>>>> Option 1: Encode the relationship in the type of buf
>>>>>
>>>>> This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
>>>>>
>>>>> Option 2: Encode the relationship in the type of size
>>>>>
>>>>> The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
>>>>>
>>>>> Thanks,
>>>>> Sid
>>>>
>>>
>
> --
> Univ.-Prof. Dr. rer. nat. Martin Uecker
> Graz University of Technology
> Institute of Biomedical Imaging
>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 10:25 ` Richard Biener
@ 2023-10-25 10:39 ` Martin Uecker
2023-10-25 18:06 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-25 10:39 UTC (permalink / raw)
To: Richard Biener
Cc: Qing Zhao, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Mittwoch, dem 25.10.2023 um 12:25 +0200 schrieb Richard Biener:
>
> > Am 25.10.2023 um 10:16 schrieb Martin Uecker <uecker@tugraz.at>:
> >
> > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > >
> > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > >
> > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > Hi, Sid,
> > > > >
> > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > >
> > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > >
> > > > > 1 #include <malloc.h>
> > > > > 2 struct A
> > > > > 3 {
> > > > > 4 size_t size;
> > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > 6 };
> > > > > 7
> > > > > 8 static size_t
> > > > > 9 get_size_from (void *ptr)
> > > > > 10 {
> > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > 12 }
> > > > > 13
> > > > > 14 void
> > > > > 15 foo (size_t sz)
> > > > > 16 {
> > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > 18 obj->size = sz;
> > > > > 19 obj->buf[0] = 2;
> > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > 21 return;
> > > > > 22 }
> > > > > 23
> > > > > 24 int main ()
> > > > > 25 {
> > > > > 26 foo (20);
> > > > > 27 return 0;
> > > > > 28 }
> > > > >
> > > > > With my GCC, it was compiled and worked:
> > > > > [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
> > > > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > > > 20
> > > > > Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
> > > > >
> > > > > [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
> > > > > [opc@qinzhao-ol8u3-x86 ]$ ./a.out
> > > > > -1
> > > > > Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
> > > > >
> > > > > Keep in mind of the above 2 situations, we will refer them in below:
> > > > >
> > > > > 1. First, the problem we are trying to resolve is:
> > > > >
> > > > > (Your description):
> > > > >
> > > > > > the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
> > > > >
> > > > > basically is correct. However, with the following exception:
> > > > >
> > > > > The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
> > > > > In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
> > > > >
> > > > > With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
> > > > >
> > > > > As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
> > > > >
> > > > > 2. Some facts I’d like to mention:
> > > > >
> > > > > A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
> > > > >
> > > > > B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
> > > > >
> > > > > 2. Then, what’s the best approach to resolve this problem:
> > > > >
> > > > > There were several suggestions so far:
> > > > >
> > > > > A. Add an additional argument, the size parameter, to __bdos,
> > > > > A.1, during FE;
> > > > > A.2, during gimplification phase;
> > > > > B. Encode the implicit USE in the type of size, to make the size “volatile”;
> > > > > C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
> > > > >
> > > > > As I explained in the above,
> > > > > ** Approach A (both A.1 and A.2) does not work;
> > > > > ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
> > > > > ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
> > > > >
> > > > > So, all the above proposed approaches, A, B, C, are not very good.
> > > > >
> > > > > Then, maybe the following might work better?
> > > > >
> > > > > In the tree optimization stage,
> > > > > * After the inlining transformation applied,
> > > > > + * Before the data-flow related optimization happens,
> > > > > + * when the data flow analysis is constructed,
> > > > >
> > > > > For each call to __bdos, add the implicit use of size parameter.
> > > > >
> > > > > Is this doable?
> > > >
> > > > Here is another proposal: Add a new builtin function
> > > >
> > > > __builtin_with_size(x, size)
> > > >
> > > > that return x but behaves similar to an allocation
> > > > function in that BDOS can look at the size argument
> > > > to discover the size.
> > > >
> > > > The FE insers this function when the field is accessed:
> > >
> > > When it’s set I suppose. Turn
> > >
> > > X.l = n;
> > >
> > > Into
> > >
> > > X.l = __builtin_with_size (x.buf, n);
> >
> > It would turn
> >
> > some_variable = (&) x.buf
> >
> > into
> >
> > some_variable = __builtin_with_size ( (&) x.buf. x.len)
>
> Unless you use the address of x.Len this will not work when len is initialized after buf. And the address will not have a meaningful data dependence.
> >
It would be a semantic requirement for this feature that
x.len needs to be initialized before x.buf is accessed.
Otherwise, I am not sure how to define the time point
at which x.len should be evaluated.
> > So the later access to x.buf and not the initialization
> > of a member of the struct (which is too early).
>
> > >
> > > And indeed we need sth like a fat pointer to reliably solve all the issues.
> >
> > What happens for other languages such as FORTRAN
> > and ADA do? Are those pointers lowered in the FE?
>
> Yes
>
> > To me it seems there are two sound ways to introduce
> > such information:
> >
> > - either by using the type system. This works in
> > the FE in C using variably modified types
> >
> > char buf[n];
> > __auto_type p = &buf;
> >
> > ... = sizeof (*p);
> >
> > But if I understand Jakob's comment to some PR
> > correctly the size information in the TREE_TYPE
> > is not processed correctly anymore in the
> > middle-end.
>
> The type based info is lowered during gimplification and in particular for pointer types the middle-end quickly loses track of the original type.
>
Would it work if we make sure that we find a suitable
type? Or in other words, are the (non-constant) size
expressions inside it still useful in later passes?
Martin
> Richard
>
> >
> > - or one injects the information via some
> > tree node or builtin at certain points in
> > time as suggested here, and the compiler
> > derives the information from these points
> > as tree-object-size does.
> >
> >
> > The use of attributes seems fragile and - looking
> > at the access attribute also overly complex. And
> > we somehow support this only for function types
> > and not elsewhere and also this then gets lost
> > during inlining. So I think for all this stuff
> > (nonnull, access, counted_by) I think a better
> > approach is needed.
> >
> >
> > Martin
> >
> >
> > >
> > > Richard
> >
> >
> >
> >
> > >
> > > > __builtin_with_size(x.buf, x.L);
> > > >
> > > >
> > > > Martin
> > > >
> > > >
> > > >
> > > > >
> > > > > Otherwise, we might need to take the “volatile” approach.
> > > > >
> > > > > Let me know your suggestion and comment.
> > > > >
> > > > > Thanks a lot.
> > > > >
> > > > > Qing
> > > > >
> > > > >
> > > > > > __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
> > > > > >
> > > > > > The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
> > > > > >
> > > > > > struct A
> > > > > > {
> > > > > > size_t size;
> > > > > > char buf[] __attribute__((counted_by(size)));
> > > > > > }
> > > > > >
> > > > > > static size_t
> > > > > > get_size_of (void *ptr)
> > > > > > {
> > > > > > return __bdos (ptr, 1);
> > > > > > }
> > > > > >
> > > > > > void
> > > > > > foo (size_t sz)
> > > > > > {
> > > > > > struct A *obj = __builtin_malloc (sz);
> > > > > > obj->size = sz;
> > > > > >
> > > > > > ...
> > > > > > __builtin_printf ("%zu\n", get_size_of (obj->array));
> > > > > > ...
> > > > > > }
> > > > > >
> > > > > > Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
> > > > > >
> > > > > > As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
> > > > > >
> > > > > > Option 1: Encode the relationship in the type of buf
> > > > > >
> > > > > > This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
> > > > > >
> > > > > > Option 2: Encode the relationship in the type of size
> > > > > >
> > > > > > The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
> > > > > >
> > > > > > Thanks,
> > > > > > Sid
> > > > >
> > > >
> >
> > --
> > Univ.-Prof. Dr. rer. nat. Martin Uecker
> > Graz University of Technology
> > Institute of Biomedical Imaging
> >
> >
--
Univ.-Prof. Dr. rer. nat. Martin Uecker
Graz University of Technology
Institute of Biomedical Imaging
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 10:25 ` Siddhesh Poyarekar
@ 2023-10-25 10:47 ` Martin Uecker
2023-10-25 11:13 ` Richard Biener
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-25 10:47 UTC (permalink / raw)
To: Siddhesh Poyarekar, Richard Biener
Cc: Qing Zhao, Joseph Myers, Jakub Jelinek, gcc Patches, kees Cook, isanbard
Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> On 2023-10-25 04:16, Martin Uecker wrote:
> > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > >
> > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > >
> > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > Hi, Sid,
> > > > >
> > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > >
> > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > >
> > > > > 1 #include <malloc.h>
> > > > > 2 struct A
> > > > > 3 {
> > > > > 4 size_t size;
> > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > 6 };
> > > > > 7
> > > > > 8 static size_t
> > > > > 9 get_size_from (void *ptr)
> > > > > 10 {
> > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > 12 }
> > > > > 13
> > > > > 14 void
> > > > > 15 foo (size_t sz)
> > > > > 16 {
> > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > 18 obj->size = sz;
> > > > > 19 obj->buf[0] = 2;
> > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > 21 return;
> > > > > 22 }
> > > > > 23
> > > > > 24 int main ()
> > > > > 25 {
> > > > > 26 foo (20);
> > > > > 27 return 0;
> > > > > 28 }
> > > > >
>
> <snip>
>
> > > When it’s set I suppose. Turn
> > >
> > > X.l = n;
> > >
> > > Into
> > >
> > > X.l = __builtin_with_size (x.buf, n);
> >
> > It would turn
> >
> > some_variable = (&) x.buf
> >
> > into
> >
> > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> >
> >
> > So the later access to x.buf and not the initialization
> > of a member of the struct (which is too early).
> >
>
> Hmm, so with Qing's example above, are you suggesting the transformation
> be to foo like so:
>
> 14 void
> 15 foo (size_t sz)
> 16 {
> 16.5 void * _1;
> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> 18 obj->size = sz;
> 19 obj->buf[0] = 2;
> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> 20 __builtin_printf (“%d\n", get_size_from (_1));
> 21 return;
> 22 }
>
> If yes then this could indeed work. I think I got thrown off by the
> reference to __bdos.
Yes. I think it is important not to evaluate the size at the
access to buf and not the allocation, because the point is to
recover it from the size member even when the compiler can't
see the original allocation.
Evaluating at this point requires that the size is correctly set
before the access to the FAM and the user has to make sure
this is the case. But to me this requirement would make sense.
Semantically, it could aöso make sense to evaluate the size at a
later time. But then the reordering becomes problematic again.
Also I think this would make this feature generally more useful.
For example, it could work also for others pointers in the struct
and not just for FAMs. In this case, the struct may already be
freed when BDOS is called, so it might also not possible to
access the size member at a later time.
Martin
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 10:47 ` Martin Uecker
@ 2023-10-25 11:13 ` Richard Biener
2023-10-25 18:16 ` Martin Uecker
2023-10-25 18:17 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Richard Biener @ 2023-10-25 11:13 UTC (permalink / raw)
To: Martin Uecker
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>
> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>
>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>
>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>> Hi, Sid,
>>>>>>
>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>
>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>
>>>>>> 1 #include <malloc.h>
>>>>>> 2 struct A
>>>>>> 3 {
>>>>>> 4 size_t size;
>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>> 6 };
>>>>>> 7
>>>>>> 8 static size_t
>>>>>> 9 get_size_from (void *ptr)
>>>>>> 10 {
>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>> 12 }
>>>>>> 13
>>>>>> 14 void
>>>>>> 15 foo (size_t sz)
>>>>>> 16 {
>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>> 18 obj->size = sz;
>>>>>> 19 obj->buf[0] = 2;
>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>> 21 return;
>>>>>> 22 }
>>>>>> 23
>>>>>> 24 int main ()
>>>>>> 25 {
>>>>>> 26 foo (20);
>>>>>> 27 return 0;
>>>>>> 28 }
>>>>>>
>>
>> <snip>
>>
>>>> When it’s set I suppose. Turn
>>>>
>>>> X.l = n;
>>>>
>>>> Into
>>>>
>>>> X.l = __builtin_with_size (x.buf, n);
>>>
>>> It would turn
>>>
>>> some_variable = (&) x.buf
>>>
>>> into
>>>
>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>
>>>
>>> So the later access to x.buf and not the initialization
>>> of a member of the struct (which is too early).
>>>
>>
>> Hmm, so with Qing's example above, are you suggesting the transformation
>> be to foo like so:
>>
>> 14 void
>> 15 foo (size_t sz)
>> 16 {
>> 16.5 void * _1;
>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>> 18 obj->size = sz;
>> 19 obj->buf[0] = 2;
>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>> 21 return;
>> 22 }
>>
>> If yes then this could indeed work. I think I got thrown off by the
>> reference to __bdos.
>
> Yes. I think it is important not to evaluate the size at the
> access to buf and not the allocation, because the point is to
> recover it from the size member even when the compiler can't
> see the original allocation.
But if the access is through a pointer without the attribute visible even the Frontend cannot recover? We’d need to force type correctness and give up on indirecting through an int * when it can refer to two diffenent container types. The best we can do I think is mark allocation sites and hope for some basic code hygiene (not clobbering size or array pointer through pointers without the appropriately attributed type)
> Evaluating at this point requires that the size is correctly set
> before the access to the FAM and the user has to make sure
> this is the case. But to me this requirement would make sense.
>
> Semantically, it could aöso make sense to evaluate the size at a
> later time. But then the reordering becomes problematic again.
>
> Also I think this would make this feature generally more useful.
> For example, it could work also for others pointers in the struct
> and not just for FAMs. In this case, the struct may already be
> freed when BDOS is called, so it might also not possible to
> access the size member at a later time.
>
> Martin
>
>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 23:56 ` Siddhesh Poyarekar
@ 2023-10-25 13:27 ` Qing Zhao
2023-10-25 14:50 ` Siddhesh Poyarekar
2023-10-25 22:06 ` Kees Cook
0 siblings, 2 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 13:27 UTC (permalink / raw)
To: Siddhesh Poyarekar, Martin Uecker
Cc: Richard Biener, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
> On Oct 24, 2023, at 7:56 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-24 18:51, Qing Zhao wrote:
>> Thanks for the proposal!
>> So what you suggested is:
>> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
>> Become:
>> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>
> Oops, I think Martin and I fell off-list in a subthread. I clarified that my comment was that any such annotation at object reference is probably too late and hence not the right place for it; basically it has the same problems as the option A in your comment. A better place to reinforce such a relationship would be the allocation+initialization site instead.
I think Martin’s proposal might work, it’s different than the option A:
A. Add an additional argument, the size parameter, to __bdos,
A.1, during FE;
A.2, during gimplification phase;
Option A targets on the __bdos call, try to encode the implicit use to the call, this will not work when the real object has not been instantiation at the call site.
However, Martin’s proposal targets on the FMA array itself, it will enhance the FAM access naturally with the size information. And such FAM access with size info will propagated to the __bdos site later through inlining, etc. and then tree-object-size can use the size information at that point. At the same time, the implicit use of the size is recorded correctly.
So, I think that this proposal is natural and reasonable.
Qing
>
> Thanks,
> Sid
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 13:27 ` Qing Zhao
@ 2023-10-25 14:50 ` Siddhesh Poyarekar
2023-10-25 15:38 ` Richard Biener
2023-10-25 18:44 ` Qing Zhao
2023-10-25 22:06 ` Kees Cook
1 sibling, 2 replies; 116+ messages in thread
From: Siddhesh Poyarekar @ 2023-10-25 14:50 UTC (permalink / raw)
To: Qing Zhao, Martin Uecker
Cc: Richard Biener, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
On 2023-10-25 09:27, Qing Zhao wrote:
>
>
>> On Oct 24, 2023, at 7:56 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>
>> On 2023-10-24 18:51, Qing Zhao wrote:
>>> Thanks for the proposal!
>>> So what you suggested is:
>>> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
>>> Become:
>>> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>>> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>>
>> Oops, I think Martin and I fell off-list in a subthread. I clarified that my comment was that any such annotation at object reference is probably too late and hence not the right place for it; basically it has the same problems as the option A in your comment. A better place to reinforce such a relationship would be the allocation+initialization site instead.
>
> I think Martin’s proposal might work, it’s different than the option A:
>
> A. Add an additional argument, the size parameter, to __bdos,
> A.1, during FE;
> A.2, during gimplification phase;
>
> Option A targets on the __bdos call, try to encode the implicit use to the call, this will not work when the real object has not been instantiation at the call site.
>
> However, Martin’s proposal targets on the FMA array itself, it will enhance the FAM access naturally with the size information. And such FAM access with size info will propagated to the __bdos site later through inlining, etc. and then tree-object-size can use the size information at that point. At the same time, the implicit use of the size is recorded correctly.
>
> So, I think that this proposal is natural and reasonable.
Ack, we discussed this later in the thread and I agree[1]. Richard
still has concerns[2] that I think may be addressed by putting
__builtin_with_size at the point where the reference to x.buf escapes,
but I'm not very sure about that.
Oh, and Martin suggested using __builtin_with_size more generally[3] in
bugzilla to address attribute inlining issues and we have high level
consensus for a __builtin_with_access instead, which associates access
type in addition to size with the target object. For the purposes of
counted_by, access type could simply be -1.
Thanks,
Sid
[1]
https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#m4f3cafa489493180e258fd62aca0196a5f244039
[2]
https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#mcf226f891621db8b640deaedd8942bb8519010f3
[3] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503#c6
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 14:50 ` Siddhesh Poyarekar
@ 2023-10-25 15:38 ` Richard Biener
2023-10-25 19:03 ` Qing Zhao
2023-10-25 18:44 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-25 15:38 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Qing Zhao, Martin Uecker, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 25.10.2023 um 16:50 schrieb Siddhesh Poyarekar <siddhesh@gotplt.org>:
>
> On 2023-10-25 09:27, Qing Zhao wrote:
>>>> On Oct 24, 2023, at 7:56 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>
>>> On 2023-10-24 18:51, Qing Zhao wrote:
>>>> Thanks for the proposal!
>>>> So what you suggested is:
>>>> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
>>>> Become:
>>>> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>>>> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>>>
>>> Oops, I think Martin and I fell off-list in a subthread. I clarified that my comment was that any such annotation at object reference is probably too late and hence not the right place for it; basically it has the same problems as the option A in your comment. A better place to reinforce such a relationship would be the allocation+initialization site instead.
>> I think Martin’s proposal might work, it’s different than the option A:
>> A. Add an additional argument, the size parameter, to __bdos,
>> A.1, during FE;
>> A.2, during gimplification phase;
>> Option A targets on the __bdos call, try to encode the implicit use to the call, this will not work when the real object has not been instantiation at the call site.
>> However, Martin’s proposal targets on the FMA array itself, it will enhance the FAM access naturally with the size information. And such FAM access with size info will propagated to the __bdos site later through inlining, etc. and then tree-object-size can use the size information at that point. At the same time, the implicit use of the size is recorded correctly.
>> So, I think that this proposal is natural and reasonable.
>
> Ack, we discussed this later in the thread and I agree[1]. Richard still has concerns[2] that I think may be addressed by putting __builtin_with_size at the point where the reference to x.buf escapes, but I'm not very sure about that.
>
> Oh, and Martin suggested using __builtin_with_size more generally[3] in bugzilla to address attribute inlining issues and we have high level consensus for a __builtin_with_access instead, which associates access type in addition to size with the target object. For the purposes of counted_by, access type could simply be -1.
Btw, I’d like to see some hard numbers on the amount of extra false positives this will cause a well as the effect on generated code before putting this in mainline and effectively needing to support it forever.
Richard
> Thanks,
> Sid
>
>
> [1] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#m4f3cafa489493180e258fd62aca0196a5f244039
>
> [2] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#mcf226f891621db8b640deaedd8942bb8519010f3
>
> [3] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503#c6
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 10:39 ` Martin Uecker
@ 2023-10-25 18:06 ` Qing Zhao
0 siblings, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 18:06 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 25, 2023, at 6:39 AM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Mittwoch, dem 25.10.2023 um 12:25 +0200 schrieb Richard Biener:
>>
>>> Am 25.10.2023 um 10:16 schrieb Martin Uecker <uecker@tugraz.at>:
>>>
>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>
>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>
>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>> Hi, Sid,
>>>>>>
>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>
>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>
>>>>>> 1 #include <malloc.h>
>>>>>> 2 struct A
>>>>>> 3 {
>>>>>> 4 size_t size;
>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>> 6 };
>>>>>> 7
>>>>>> 8 static size_t
>>>>>> 9 get_size_from (void *ptr)
>>>>>> 10 {
>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>> 12 }
>>>>>> 13
>>>>>> 14 void
>>>>>> 15 foo (size_t sz)
>>>>>> 16 {
>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>> 18 obj->size = sz;
>>>>>> 19 obj->buf[0] = 2;
>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>> 21 return;
>>>>>> 22 }
>>>>>> 23
>>>>>> 24 int main ()
>>>>>> 25 {
>>>>>> 26 foo (20);
>>>>>> 27 return 0;
>>>>>> 28 }
>>>>>>
>>>>>> With my GCC, it was compiled and worked:
>>>>>> [opc@qinzhao-ol8u3-x86 ]$ /home/opc/Install/latest-d/bin/gcc -O1 t5.c
>>>>>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>>>>>> 20
>>>>>> Situation 1: With O1 and above, the routine “get_size_from” was inlined into “foo”, therefore, the call to __bdos is in the same routine as the instantiation of the object, and the TYPE information and the attached counted_by attribute information in the TYPE of the object can be USED by the __bdos call to compute the final object size.
>>>>>>
>>>>>> [opc@qinzhao-ol8u3-x86]$ /home/opc/Install/latest-d/bin/gcc -O0 t5.c
>>>>>> [opc@qinzhao-ol8u3-x86 ]$ ./a.out
>>>>>> -1
>>>>>> Situation 2: With O0, the routine “get_size_from” was NOT inlined into “foo”, therefore, the call to __bdos is Not in the same routine as the instantiation of the object, As a result, the TYPE info and the attached counted_by info of the object can NOT be USED by the __bdos call.
>>>>>>
>>>>>> Keep in mind of the above 2 situations, we will refer them in below:
>>>>>>
>>>>>> 1. First, the problem we are trying to resolve is:
>>>>>>
>>>>>> (Your description):
>>>>>>
>>>>>>> the reordering of __bdos w.r.t. initialization of the size parameter but to also account for DSE of the assignment, we can abstract this problem to that of DFA being unable to see implicit use of the size parameter in the __bdos call.
>>>>>>
>>>>>> basically is correct. However, with the following exception:
>>>>>>
>>>>>> The implicit use of the size parameter in the __bdos call is not always there, it ONLY exists WHEN the __bdos is able to evaluated to an expression of the size parameter in the “objsz” phase, i.e., the “Situation 1” of the above example.
>>>>>> In the “Situation 2”, when the __bdos does not see the TYPE of the real object, it does not see the counted_by information from the TYPE, therefore, it is not able to evaluate the size of the object through the counted_by information. As a result, the implicit use of the size parameter in the __bdos call does NOT exist at all. The optimizer can freely reorder the initialization of the size parameter with the __bdos call since there is no data flow dependency between these two.
>>>>>>
>>>>>> With this exception in mind, we can see that your proposed “option 2” (making the type of size “volatile”) is too conservative, it will disable many optimizations unnecessarily, even though it’s safe and simple to implement.
>>>>>>
>>>>>> As a compiler optimization person for many many years, I really don’t want to take this approach at this moment. -:)
>>>>>>
>>>>>> 2. Some facts I’d like to mention:
>>>>>>
>>>>>> A. The incorrect reordering (or CSE) potential ONLY exists in the TREE optimization stage. During RTL stage, the __bdos call has already been replaced by an expression of the size parameter or a constant, the data dependency is explicitly in the IR already. I believe that the data analysis in RTL stage should pick up the data dependency correctly, No special handling is needed in RTL.
>>>>>>
>>>>>> B. If the __bdos call cannot see the real object , it has no way to get the “counted_by” field from the TYPE of the real object. So, if we try to add the implicit use of the “counted_by” field to the __bdos call, the object instantiation should be in the same routine as the __bdos call. Both the FE and the gimplification phase are too early to do this work.
>>>>>>
>>>>>> 2. Then, what’s the best approach to resolve this problem:
>>>>>>
>>>>>> There were several suggestions so far:
>>>>>>
>>>>>> A. Add an additional argument, the size parameter, to __bdos,
>>>>>> A.1, during FE;
>>>>>> A.2, during gimplification phase;
>>>>>> B. Encode the implicit USE in the type of size, to make the size “volatile”;
>>>>>> C. Encode the implicit USE in the type of buf, then update the optimization passes to use this implicit USE encoded in the type of buf.
>>>>>>
>>>>>> As I explained in the above,
>>>>>> ** Approach A (both A.1 and A.2) does not work;
>>>>>> ** Approach B will have big performance impact, I’d prefer not to take this approach at this moment.
>>>>>> ** Approach C will be a lot of change in GCC, and also not very necessary since the ONLY implicit use of the size parameter is in the __bdos call when __bdos can see the real object.
>>>>>>
>>>>>> So, all the above proposed approaches, A, B, C, are not very good.
>>>>>>
>>>>>> Then, maybe the following might work better?
>>>>>>
>>>>>> In the tree optimization stage,
>>>>>> * After the inlining transformation applied,
>>>>>> + * Before the data-flow related optimization happens,
>>>>>> + * when the data flow analysis is constructed,
>>>>>>
>>>>>> For each call to __bdos, add the implicit use of size parameter.
>>>>>>
>>>>>> Is this doable?
>>>>>
>>>>> Here is another proposal: Add a new builtin function
>>>>>
>>>>> __builtin_with_size(x, size)
>>>>>
>>>>> that return x but behaves similar to an allocation
>>>>> function in that BDOS can look at the size argument
>>>>> to discover the size.
>>>>>
>>>>> The FE insers this function when the field is accessed:
>>>>
>>>> When it’s set I suppose. Turn
>>>>
>>>> X.l = n;
>>>>
>>>> Into
>>>>
>>>> X.l = __builtin_with_size (x.buf, n);
>>>
>>> It would turn
>>>
>>> some_variable = (&) x.buf
>>>
>>> into
>>>
>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>
>> Unless you use the address of x.Len this will not work when len is initialized after buf. And the address will not have a meaningful data dependence.
>>>
>
> It would be a semantic requirement for this feature that
> x.len needs to be initialized before x.buf is accessed.
Yes, that’s right, we might need to clarify this into the documentation of the counted_by.
It should be a user error if the source code violate this rule.
Qing
>
> Otherwise, I am not sure how to define the time point
> at which x.len should be evaluated.
>
>>> So the later access to x.buf and not the initialization
>>> of a member of the struct (which is too early).
>>
>>>>
>>>> And indeed we need sth like a fat pointer to reliably solve all the issues.
>>>
>>> What happens for other languages such as FORTRAN
>>> and ADA do? Are those pointers lowered in the FE?
>>
>> Yes
>>
>>> To me it seems there are two sound ways to introduce
>>> such information:
>>>
>>> - either by using the type system. This works in
>>> the FE in C using variably modified types
>>>
>>> char buf[n];
>>> __auto_type p = &buf;
>>>
>>> ... = sizeof (*p);
>>>
>>> But if I understand Jakob's comment to some PR
>>> correctly the size information in the TREE_TYPE
>>> is not processed correctly anymore in the
>>> middle-end.
>>
>> The type based info is lowered during gimplification and in particular for pointer types the middle-end quickly loses track of the original type.
>>
>
> Would it work if we make sure that we find a suitable
> type? Or in other words, are the (non-constant) size
> expressions inside it still useful in later passes?
>
> Martin
>
>
>> Richard
>>
>>>
>>> - or one injects the information via some
>>> tree node or builtin at certain points in
>>> time as suggested here, and the compiler
>>> derives the information from these points
>>> as tree-object-size does.
>>>
>>>
>>> The use of attributes seems fragile and - looking
>>> at the access attribute also overly complex. And
>>> we somehow support this only for function types
>>> and not elsewhere and also this then gets lost
>>> during inlining. So I think for all this stuff
>>> (nonnull, access, counted_by) I think a better
>>> approach is needed.
>>>
>>>
>>> Martin
>>>
>>>
>>>>
>>>> Richard
>>>
>>>
>>>
>>>
>>>>
>>>>> __builtin_with_size(x.buf, x.L);
>>>>>
>>>>>
>>>>> Martin
>>>>>
>>>>>
>>>>>
>>>>>>
>>>>>> Otherwise, we might need to take the “volatile” approach.
>>>>>>
>>>>>> Let me know your suggestion and comment.
>>>>>>
>>>>>> Thanks a lot.
>>>>>>
>>>>>> Qing
>>>>>>
>>>>>>
>>>>>>> __bdos is the one such implicit user of the size parameter and you're proposing to solve this by encoding the relationship between buffer and size at the __bdos call site. But what about the case when the instantiation of the object is not at the same place as the __bdos call site, i.e. the DFA is unable to make that relationship?
>>>>>>>
>>>>>>> The example Martin showed where the subobject gets "hidden" behind a pointer was a trivial one where DFA *may* actually work in practice (because the object-size pass can thread through these assignments) but think about this one:
>>>>>>>
>>>>>>> struct A
>>>>>>> {
>>>>>>> size_t size;
>>>>>>> char buf[] __attribute__((counted_by(size)));
>>>>>>> }
>>>>>>>
>>>>>>> static size_t
>>>>>>> get_size_of (void *ptr)
>>>>>>> {
>>>>>>> return __bdos (ptr, 1);
>>>>>>> }
>>>>>>>
>>>>>>> void
>>>>>>> foo (size_t sz)
>>>>>>> {
>>>>>>> struct A *obj = __builtin_malloc (sz);
>>>>>>> obj->size = sz;
>>>>>>>
>>>>>>> ...
>>>>>>> __builtin_printf ("%zu\n", get_size_of (obj->array));
>>>>>>> ...
>>>>>>> }
>>>>>>>
>>>>>>> Until get_size_of is inlined, no DFA can see the __bdos call in the same place as the point where obj is allocated. As a result, the assignment to obj->size could get reordered (or the store eliminated) w.r.t. the __bdos call until the inlining happens.
>>>>>>>
>>>>>>> As a result, the relationship between buf and size established by the attribute needs to be encoded into the type somehow. There are two options:
>>>>>>>
>>>>>>> Option 1: Encode the relationship in the type of buf
>>>>>>>
>>>>>>> This is kinda what you end up doing with component_ref_has_counted_by and it does show the relationship if one is looking (through that call), but nothing more that can be used to, e.g. prevent reordering or tell the optimizer that the reference to the buf member may imply a reference to the size member as well. This could be remedied by somehow encoding the USES relationship for size into the type of buf that the optimization passes can see. I feel like this may be a bit convoluted to specify in a future language extension in a way that will actually be well understood by developers, but it will likely generate faster runtime code. This will also likely require a bigger change across passes.
>>>>>>>
>>>>>>> Option 2: Encode the relationship in the type of size
>>>>>>>
>>>>>>> The other option is to enhance the type of size somehow so that it discourages reordering and store elimination, basically pessimizing code. I think volatile semantics might be the way to do this and may even be straightforward to specify in the future language extension given that it builds on a known language construct and is thematically related. However it does pessimize output for code that implements __counted_by__.
>>>>>>>
>>>>>>> Thanks,
>>>>>>> Sid
>>>>>>
>>>>>
>>>
>>> --
>>> Univ.-Prof. Dr. rer. nat. Martin Uecker
>>> Graz University of Technology
>>> Institute of Biomedical Imaging
>>>
>>>
>
> --
> Univ.-Prof. Dr. rer. nat. Martin Uecker
> Graz University of Technology
> Institute of Biomedical Imaging
>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 11:13 ` Richard Biener
@ 2023-10-25 18:16 ` Martin Uecker
2023-10-26 8:45 ` Richard Biener
2023-10-25 18:17 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-25 18:16 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>
> > Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
> >
> > Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> > > > On 2023-10-25 04:16, Martin Uecker wrote:
> > > > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > > > >
> > > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > >
> > > > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > > > Hi, Sid,
> > > > > > >
> > > > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > > > >
> > > > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > > > >
> > > > > > > 1 #include <malloc.h>
> > > > > > > 2 struct A
> > > > > > > 3 {
> > > > > > > 4 size_t size;
> > > > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > > > 6 };
> > > > > > > 7
> > > > > > > 8 static size_t
> > > > > > > 9 get_size_from (void *ptr)
> > > > > > > 10 {
> > > > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > > > 12 }
> > > > > > > 13
> > > > > > > 14 void
> > > > > > > 15 foo (size_t sz)
> > > > > > > 16 {
> > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > 18 obj->size = sz;
> > > > > > > 19 obj->buf[0] = 2;
> > > > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > > > 21 return;
> > > > > > > 22 }
> > > > > > > 23
> > > > > > > 24 int main ()
> > > > > > > 25 {
> > > > > > > 26 foo (20);
> > > > > > > 27 return 0;
> > > > > > > 28 }
> > > > > > >
> > >
> > > <snip>
> > >
> > > > > When it’s set I suppose. Turn
> > > > >
> > > > > X.l = n;
> > > > >
> > > > > Into
> > > > >
> > > > > X.l = __builtin_with_size (x.buf, n);
> > > >
> > > > It would turn
> > > >
> > > > some_variable = (&) x.buf
> > > >
> > > > into
> > > >
> > > > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> > > >
> > > >
> > > > So the later access to x.buf and not the initialization
> > > > of a member of the struct (which is too early).
> > > >
> > >
> > > Hmm, so with Qing's example above, are you suggesting the transformation
> > > be to foo like so:
> > >
> > > 14 void
> > > 15 foo (size_t sz)
> > > 16 {
> > > 16.5 void * _1;
> > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > 18 obj->size = sz;
> > > 19 obj->buf[0] = 2;
> > > 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> > > 20 __builtin_printf (“%d\n", get_size_from (_1));
> > > 21 return;
> > > 22 }
> > >
> > > If yes then this could indeed work. I think I got thrown off by the
> > > reference to __bdos.
> >
> > Yes. I think it is important not to evaluate the size at the
> > access to buf and not the allocation, because the point is to
> > recover it from the size member even when the compiler can't
> > see the original allocation.
>
> But if the access is through a pointer without the attribute visible
> even the Frontend cannot recover?
Yes, if the access is using a struct-with-FAM without the attribute
the FE would not be insert the builtin. BDOS could potentially
still see the original allocation but if it doesn't, then there is
no information.
> We’d need to force type correctness and give up on indirecting
> through an int * when it can refer to two diffenent container types.
> The best we can do I think is mark allocation sites and hope for
> some basic code hygiene (not clobbering size or array pointer
> through pointers without the appropriately attributed type)
I am do not fully understand what you are referring to. But yes,
for full bounds safety we would need the language feature.
In C people should start to variably-modified types
more. I think we can build perfect bounds safety on top of
them in a very good way with only FE changes.
All these attributes are just a best effort. But for a while,
this will be necessary.
Martin
>
> > Evaluating at this point requires that the size is correctly set
> > before the access to the FAM and the user has to make sure
> > this is the case. But to me this requirement would make sense.
> >
> > Semantically, it could aöso make sense to evaluate the size at a
> > later time. But then the reordering becomes problematic again.
> >
> > Also I think this would make this feature generally more useful.
> > For example, it could work also for others pointers in the struct
> > and not just for FAMs. In this case, the struct may already be
> > freed when BDOS is called, so it might also not possible to
> > access the size member at a later time.
> >
> > Martin
> >
> >
> > >
> >
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 11:13 ` Richard Biener
2023-10-25 18:16 ` Martin Uecker
@ 2023-10-25 18:17 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 18:17 UTC (permalink / raw)
To: Richard Biener
Cc: Martin Uecker, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 25, 2023, at 7:13 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
>
>
>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>
>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>
>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>
>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>> Hi, Sid,
>>>>>>>
>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>
>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>
>>>>>>> 1 #include <malloc.h>
>>>>>>> 2 struct A
>>>>>>> 3 {
>>>>>>> 4 size_t size;
>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>> 6 };
>>>>>>> 7
>>>>>>> 8 static size_t
>>>>>>> 9 get_size_from (void *ptr)
>>>>>>> 10 {
>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>> 12 }
>>>>>>> 13
>>>>>>> 14 void
>>>>>>> 15 foo (size_t sz)
>>>>>>> 16 {
>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>> 18 obj->size = sz;
>>>>>>> 19 obj->buf[0] = 2;
>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>> 21 return;
>>>>>>> 22 }
>>>>>>> 23
>>>>>>> 24 int main ()
>>>>>>> 25 {
>>>>>>> 26 foo (20);
>>>>>>> 27 return 0;
>>>>>>> 28 }
>>>>>>>
>>>
>>> <snip>
>>>
>>>>> When it’s set I suppose. Turn
>>>>>
>>>>> X.l = n;
>>>>>
>>>>> Into
>>>>>
>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>
>>>> It would turn
>>>>
>>>> some_variable = (&) x.buf
>>>>
>>>> into
>>>>
>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>
>>>>
>>>> So the later access to x.buf and not the initialization
>>>> of a member of the struct (which is too early).
>>>>
>>>
>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>> be to foo like so:
>>>
>>> 14 void
>>> 15 foo (size_t sz)
>>> 16 {
>>> 16.5 void * _1;
>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>> 18 obj->size = sz;
>>> 19 obj->buf[0] = 2;
>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>> 21 return;
>>> 22 }
>>>
>>> If yes then this could indeed work. I think I got thrown off by the
>>> reference to __bdos.
>>
>> Yes. I think it is important not to evaluate the size at the
>> access to buf and not the allocation, because the point is to
>> recover it from the size member even when the compiler can't
>> see the original allocation.
>
> But if the access is through a pointer without the attribute visible even the Frontend cannot recover? We’d need to force type correctness and give up on indirecting through an int * when it can refer to two diffenent container types.
Might need issue warnings when this happens?
> The best we can do I think is mark allocation sites and hope for some basic code hygiene (not clobbering size or array pointer through pointers without the appropriately attributed type)
I guess that we need to clarify the requirement in the documentation, and also issue warnings when the source code has such issues.
Qing
>
>> Evaluating at this point requires that the size is correctly set
>> before the access to the FAM and the user has to make sure
>> this is the case. But to me this requirement would make sense.
>>
>> Semantically, it could aöso make sense to evaluate the size at a
>> later time. But then the reordering becomes problematic again.
>>
>> Also I think this would make this feature generally more useful.
>> For example, it could work also for others pointers in the struct
>> and not just for FAMs. In this case, the struct may already be
>> freed when BDOS is called, so it might also not possible to
>> access the size member at a later time.
>>
>> Martin
>>
>>
>>>
>>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 14:50 ` Siddhesh Poyarekar
2023-10-25 15:38 ` Richard Biener
@ 2023-10-25 18:44 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 18:44 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Martin Uecker, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 25, 2023, at 10:50 AM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>
> On 2023-10-25 09:27, Qing Zhao wrote:
>>> On Oct 24, 2023, at 7:56 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>
>>> On 2023-10-24 18:51, Qing Zhao wrote:
>>>> Thanks for the proposal!
>>>> So what you suggested is:
>>>> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
>>>> Become:
>>>> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>>>> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>>>
>>> Oops, I think Martin and I fell off-list in a subthread. I clarified that my comment was that any such annotation at object reference is probably too late and hence not the right place for it; basically it has the same problems as the option A in your comment. A better place to reinforce such a relationship would be the allocation+initialization site instead.
>> I think Martin’s proposal might work, it’s different than the option A:
>> A. Add an additional argument, the size parameter, to __bdos,
>> A.1, during FE;
>> A.2, during gimplification phase;
>> Option A targets on the __bdos call, try to encode the implicit use to the call, this will not work when the real object has not been instantiation at the call site.
>> However, Martin’s proposal targets on the FMA array itself, it will enhance the FAM access naturally with the size information. And such FAM access with size info will propagated to the __bdos site later through inlining, etc. and then tree-object-size can use the size information at that point. At the same time, the implicit use of the size is recorded correctly.
>> So, I think that this proposal is natural and reasonable.
>
> Ack, we discussed this later in the thread and I agree[1]. Richard still has concerns[2] that I think may be addressed by putting __builtin_with_size at the point where the reference to x.buf escapes, but I'm not very sure about that.
>
> Oh, and Martin suggested using __builtin_with_size more generally[3] in bugzilla to address attribute inlining issues and we have high level consensus for a __builtin_with_access instead, which associates access type in addition to size with the target object. For the purposes of counted_by, access type could simply be -1.
Yes, I read all the discussions in the comments of PR96503 (https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503), and I do agree that this is a good idea.
I prefer the name for the new builtin as:
__builtin_with_access_and_size
Instead of
__builtin_with_access
All the attributes, “alloca_size”, “access”, and the new “counted_by” for FMA, could be converted to this builtin consistently, and even the later new extension, for example, “counted_by” attribute for general pointers, could use the same builtin.
SOMETYPE *ptr = __builtin_with_access_and_size (SOMETYPE *ptr, size_t size, int access)
In the above,
1. SOMETYPE will be the type of the pointee of “ptr”, it could be a real type or void.
2. “size”
If SOMETYPE is a real type, the “size” will be the number of elements of the type;
If SOMETYPE is void, the “size” will be the number of bytes.
3. “access”
-1: Unknown access semantics
0: none
1: read_only
2: write_only
3: read_write
For the “counted_by” and “alloca_size” attribute, the “access” will be -1.
Qing
>
> Thanks,
> Sid
>
>
> [1] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#m4f3cafa489493180e258fd62aca0196a5f244039
>
> [2] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#mcf226f891621db8b640deaedd8942bb8519010f3
>
> [3] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503#c6
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 15:38 ` Richard Biener
@ 2023-10-25 19:03 ` Qing Zhao
2023-10-26 5:21 ` Jakub Jelinek
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 19:03 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Martin Uecker, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 25, 2023, at 11:38 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
>
>
>> Am 25.10.2023 um 16:50 schrieb Siddhesh Poyarekar <siddhesh@gotplt.org>:
>>
>> On 2023-10-25 09:27, Qing Zhao wrote:
>>>>> On Oct 24, 2023, at 7:56 PM, Siddhesh Poyarekar <siddhesh@gotplt.org> wrote:
>>>>
>>>> On 2023-10-24 18:51, Qing Zhao wrote:
>>>>> Thanks for the proposal!
>>>>> So what you suggested is:
>>>>> For every x.buf, change it as a __builtin_with_size(x.buf, x.L) in the FE, then the call to the _bdos (x.buf, 1) will
>>>>> Become:
>>>>> _bdos(__builtin_with_size(x.buf, x.L), 1)?
>>>>> Then the implicit use of x.L in _bdos(x.buf.1) will become explicit?
>>>>
>>>> Oops, I think Martin and I fell off-list in a subthread. I clarified that my comment was that any such annotation at object reference is probably too late and hence not the right place for it; basically it has the same problems as the option A in your comment. A better place to reinforce such a relationship would be the allocation+initialization site instead.
>>> I think Martin’s proposal might work, it’s different than the option A:
>>> A. Add an additional argument, the size parameter, to __bdos,
>>> A.1, during FE;
>>> A.2, during gimplification phase;
>>> Option A targets on the __bdos call, try to encode the implicit use to the call, this will not work when the real object has not been instantiation at the call site.
>>> However, Martin’s proposal targets on the FMA array itself, it will enhance the FAM access naturally with the size information. And such FAM access with size info will propagated to the __bdos site later through inlining, etc. and then tree-object-size can use the size information at that point. At the same time, the implicit use of the size is recorded correctly.
>>> So, I think that this proposal is natural and reasonable.
>>
>> Ack, we discussed this later in the thread and I agree[1]. Richard still has concerns[2] that I think may be addressed by putting __builtin_with_size at the point where the reference to x.buf escapes, but I'm not very sure about that.
>>
>> Oh, and Martin suggested using __builtin_with_size more generally[3] in bugzilla to address attribute inlining issues and we have high level consensus for a __builtin_with_access instead, which associates access type in addition to size with the target object. For the purposes of counted_by, access type could simply be -1.
>
> Btw, I’d like to see some hard numbers on the amount of extra false positives this will cause a well as the effect on generated code before putting this in mainline and effectively needing to support it forever.
What do you mean by the “extra false positives”?
For the code generation impact:
turning the original x.buf
to a builtin function call
__builtin_with_access_and_size(x,buf, x.L,-1)
might inhibit some optimizations from happening before the builtin is evaluated into object size info (phase .objsz1). I guess there might be some performance impact.
However, if we mark this builtin as PURE, NOTRROW, etc, then the negative performance impact will be reduced to minimum?
Qing
>
> Richard
>
>> Thanks,
>> Sid
>>
>>
>> [1] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#m4f3cafa489493180e258fd62aca0196a5f244039
>>
>> [2] https://inbox.sourceware.org/gcc-patches/73af949c-3caa-4b11-93ce-3064b95a9908@gotplt.org/T/#mcf226f891621db8b640deaedd8942bb8519010f3
>>
>> [3] https://gcc.gnu.org/bugzilla/show_bug.cgi?id=96503#c6
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-24 23:51 ` Siddhesh Poyarekar
@ 2023-10-25 21:59 ` Kees Cook
0 siblings, 0 replies; 116+ messages in thread
From: Kees Cook @ 2023-10-25 21:59 UTC (permalink / raw)
To: Siddhesh Poyarekar
Cc: Qing Zhao, Richard Biener, Martin Uecker, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
On Tue, Oct 24, 2023 at 07:51:55PM -0400, Siddhesh Poyarekar wrote:
> Yes, that's the tradeoff. However, maybe this is the point where Kees jumps
> in and say the kernel doesn't really care as much or something like that :)
"I only care about -O2" :)
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 13:27 ` Qing Zhao
2023-10-25 14:50 ` Siddhesh Poyarekar
@ 2023-10-25 22:06 ` Kees Cook
2023-10-25 22:27 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-25 22:06 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Martin Uecker, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
On Wed, Oct 25, 2023 at 01:27:29PM +0000, Qing Zhao wrote:
> A. Add an additional argument, the size parameter, to __bdos,
> A.1, during FE;
> A.2, during gimplification phase;
I just wanted to clarify that this is all just an "internal" detail,
yes? i.e. the __bdos() used by in C code is unchanged?
For example, the Linux kernel can still use __bdos() without knowing
the count member ahead of time (otherwise it kind of defeats the purpose).
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 22:06 ` Kees Cook
@ 2023-10-25 22:27 ` Qing Zhao
2023-10-25 22:32 ` Kees Cook
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-25 22:27 UTC (permalink / raw)
To: Kees Cook
Cc: Siddhesh Poyarekar, Martin Uecker, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
> On Oct 25, 2023, at 6:06 PM, Kees Cook <keescook@chromium.org> wrote:
>
> On Wed, Oct 25, 2023 at 01:27:29PM +0000, Qing Zhao wrote:
>> A. Add an additional argument, the size parameter, to __bdos,
>> A.1, during FE;
>> A.2, during gimplification phase;
>
> I just wanted to clarify that this is all just an "internal" detail,
> yes?
YES!
> i.e. the __bdos() used by in C code is unchanged?
there should be no change to the user interface.
>
> For example, the Linux kernel can still use __bdos() without knowing
> the count member ahead of time (otherwise it kind of defeats the purpose).
Don’t quite understand this, could you clarify?
(Anyway, the bottom line is no change to the user interface, we just discuss the internal implementation inside GCC) -:)
Qing
>
> --
> Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 22:27 ` Qing Zhao
@ 2023-10-25 22:32 ` Kees Cook
2023-10-26 8:15 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-25 22:32 UTC (permalink / raw)
To: Qing Zhao
Cc: Siddhesh Poyarekar, Martin Uecker, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
On Wed, Oct 25, 2023 at 10:27:41PM +0000, Qing Zhao wrote:
>
>
> > On Oct 25, 2023, at 6:06 PM, Kees Cook <keescook@chromium.org> wrote:
> >
> > On Wed, Oct 25, 2023 at 01:27:29PM +0000, Qing Zhao wrote:
> >> A. Add an additional argument, the size parameter, to __bdos,
> >> A.1, during FE;
> >> A.2, during gimplification phase;
> >
> > I just wanted to clarify that this is all just an "internal" detail,
> > yes?
>
> YES!
Okay, I thought so, but I just wanted to double-check. :)
> > For example, the Linux kernel can still use __bdos() without knowing
> > the count member ahead of time (otherwise it kind of defeats the purpose).
> Don’t quite understand this, could you clarify?
I was just trying to explain why a chance would be a problem. But it
doesn't matter, so nevermind. :)
> (Anyway, the bottom line is no change to the user interface, we just discuss the internal implementation inside GCC) -:)
Great! I'll go back to lurking. :)
Thanks!
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 19:03 ` Qing Zhao
@ 2023-10-26 5:21 ` Jakub Jelinek
2023-10-26 8:56 ` Richard Biener
2023-10-26 14:41 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Jakub Jelinek @ 2023-10-26 5:21 UTC (permalink / raw)
To: Qing Zhao
Cc: Richard Biener, Siddhesh Poyarekar, Martin Uecker, Joseph Myers,
gcc Patches, kees Cook, isanbard
On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
> For the code generation impact:
>
> turning the original x.buf
> to a builtin function call
> __builtin_with_access_and_size(x,buf, x.L,-1)
>
> might inhibit some optimizations from happening before the builtin is
> evaluated into object size info (phase .objsz1). I guess there might be
> some performance impact.
>
> However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
> performance impact will be reduced to minimum?
You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
be able to figure out the dynamic sizes in case of normal (non-early)
inlining - caller takes address of a counted_by array, passes it down
to callee which is only inlined late and uses __bdos, or callee takes address
and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
And while the builtin (or if it is an internal detail rather than user
accessible builtin an internal function) could be even const/nothrow/leaf if
the arguments contain the loads from the structure 2 fields, I'm afraid it
will still have huge code generation impact, prevent tons of pre-IPA
optimizations. And it will need some work to handle it properly during
inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
values, so it wouldn't be just the builtin/internal-fn call to be ignored,
but also the count load from memory.
Jakub
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 22:32 ` Kees Cook
@ 2023-10-26 8:15 ` Martin Uecker
2023-10-26 16:13 ` Kees Cook
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 8:15 UTC (permalink / raw)
To: Kees Cook, Qing Zhao
Cc: Siddhesh Poyarekar, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, isanbard
Am Mittwoch, dem 25.10.2023 um 15:32 -0700 schrieb Kees Cook:
> On Wed, Oct 25, 2023 at 10:27:41PM +0000, Qing Zhao wrote:
> >
> >
> > > On Oct 25, 2023, at 6:06 PM, Kees Cook <keescook@chromium.org> wrote:
> > >
> > > On Wed, Oct 25, 2023 at 01:27:29PM +0000, Qing Zhao wrote:
> > > > A. Add an additional argument, the size parameter, to __bdos,
> > > > A.1, during FE;
> > > > A.2, during gimplification phase;
> > >
> > > I just wanted to clarify that this is all just an "internal" detail,
> > > yes?
> >
> > YES!
>
> Okay, I thought so, but I just wanted to double-check. :)
>
> > > For example, the Linux kernel can still use __bdos() without knowing
> > > the count member ahead of time (otherwise it kind of defeats the purpose).
> > Don’t quite understand this, could you clarify?
>
> I was just trying to explain why a chance would be a problem. But it
> doesn't matter, so nevermind. :)
>
> > (Anyway, the bottom line is no change to the user interface, we just discuss the internal implementation inside GCC) -:)
>
> Great! I'll go back to lurking. :)
>
> Thanks!
>
While it is about the internal implementation, it would
potentially affect the semantics of the attribute:
This would work:
x->count = 10;
char *p = &x->buf;
but not this:
char *p = &x->buf;
x->count = 1;
p[10] = 1; // !
(because the pointer is passed around the
store to the counter)
and also here the second store is then irrelevant
for the access:
x->count = 10;
char* p = &x->buf;
...
x->count = 1; // somewhere else
----
p[9] = 1; // ok, because count matter when buf was accesssed.
IMHO this makes sense also from the user side and
are the desirable semantics we discussed before.
But can you take a look at this?
This should simulate it fairly well:
https://godbolt.org/z/xq89aM7Gr
(the call to the noinline function would go away,
but not necessarily its impact on optimization)
Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-25 18:16 ` Martin Uecker
@ 2023-10-26 8:45 ` Richard Biener
2023-10-26 9:20 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-26 8:45 UTC (permalink / raw)
To: Martin Uecker
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
> >
> > > Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
> > >
> > > Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> > > > > On 2023-10-25 04:16, Martin Uecker wrote:
> > > > > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > > > > >
> > > > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > > >
> > > > > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > > > > Hi, Sid,
> > > > > > > >
> > > > > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > > > > >
> > > > > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > > > > >
> > > > > > > > 1 #include <malloc.h>
> > > > > > > > 2 struct A
> > > > > > > > 3 {
> > > > > > > > 4 size_t size;
> > > > > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > > > > 6 };
> > > > > > > > 7
> > > > > > > > 8 static size_t
> > > > > > > > 9 get_size_from (void *ptr)
> > > > > > > > 10 {
> > > > > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > > > > 12 }
> > > > > > > > 13
> > > > > > > > 14 void
> > > > > > > > 15 foo (size_t sz)
> > > > > > > > 16 {
> > > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > > 18 obj->size = sz;
> > > > > > > > 19 obj->buf[0] = 2;
> > > > > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > > > > 21 return;
> > > > > > > > 22 }
> > > > > > > > 23
> > > > > > > > 24 int main ()
> > > > > > > > 25 {
> > > > > > > > 26 foo (20);
> > > > > > > > 27 return 0;
> > > > > > > > 28 }
> > > > > > > >
> > > >
> > > > <snip>
> > > >
> > > > > > When it’s set I suppose. Turn
> > > > > >
> > > > > > X.l = n;
> > > > > >
> > > > > > Into
> > > > > >
> > > > > > X.l = __builtin_with_size (x.buf, n);
> > > > >
> > > > > It would turn
> > > > >
> > > > > some_variable = (&) x.buf
> > > > >
> > > > > into
> > > > >
> > > > > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> > > > >
> > > > >
> > > > > So the later access to x.buf and not the initialization
> > > > > of a member of the struct (which is too early).
> > > > >
> > > >
> > > > Hmm, so with Qing's example above, are you suggesting the transformation
> > > > be to foo like so:
> > > >
> > > > 14 void
> > > > 15 foo (size_t sz)
> > > > 16 {
> > > > 16.5 void * _1;
> > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > 18 obj->size = sz;
> > > > 19 obj->buf[0] = 2;
> > > > 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> > > > 20 __builtin_printf (“%d\n", get_size_from (_1));
> > > > 21 return;
> > > > 22 }
> > > >
> > > > If yes then this could indeed work. I think I got thrown off by the
> > > > reference to __bdos.
> > >
> > > Yes. I think it is important not to evaluate the size at the
> > > access to buf and not the allocation, because the point is to
> > > recover it from the size member even when the compiler can't
> > > see the original allocation.
> >
> > But if the access is through a pointer without the attribute visible
> > even the Frontend cannot recover?
>
> Yes, if the access is using a struct-with-FAM without the attribute
> the FE would not be insert the builtin. BDOS could potentially
> still see the original allocation but if it doesn't, then there is
> no information.
>
> > We’d need to force type correctness and give up on indirecting
> > through an int * when it can refer to two diffenent container types.
> > The best we can do I think is mark allocation sites and hope for
> > some basic code hygiene (not clobbering size or array pointer
> > through pointers without the appropriately attributed type)
>
> I am do not fully understand what you are referring to.
struct A { int n; int data[n]; };
struct B { long n; int data[n]; };
int *p = flag ? a->data : b->data;
access *p;
Since we need to allow interoperability of pointers (a->data is
convertible to a non-fat pointer of type int *) this leaves us with
ambiguity we need to conservatively handle to avoid false positives.
We _might_ want to diagnose decay of a->data to int *, but IIRC
there's no way (or proposal) to allow declaring a corresponding
fat pointer, so it's not a good designed feature.
Having __builtin_with_size at allocation would possibly make
the BOS use-def walk discover both objects. I think you can't
insert __builtin_with_size at the access to *p, but in practice
that would be very much needed.
Richard.
> But yes,
> for full bounds safety we would need the language feature.
> In C people should start to variably-modified types
> more. I think we can build perfect bounds safety on top of
> them in a very good way with only FE changes.
>
> All these attributes are just a best effort. But for a while,
> this will be necessary.
>
> Martin
>
> >
> > > Evaluating at this point requires that the size is correctly set
> > > before the access to the FAM and the user has to make sure
> > > this is the case. But to me this requirement would make sense.
> > >
> > > Semantically, it could aöso make sense to evaluate the size at a
> > > later time. But then the reordering becomes problematic again.
> > >
> > > Also I think this would make this feature generally more useful.
> > > For example, it could work also for others pointers in the struct
> > > and not just for FAMs. In this case, the struct may already be
> > > freed when BDOS is called, so it might also not possible to
> > > access the size member at a later time.
> > >
> > > Martin
> > >
> > >
> > > >
> > >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 5:21 ` Jakub Jelinek
@ 2023-10-26 8:56 ` Richard Biener
2023-10-26 14:58 ` Qing Zhao
2023-10-26 14:41 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-26 8:56 UTC (permalink / raw)
To: Jakub Jelinek
Cc: Qing Zhao, Siddhesh Poyarekar, Martin Uecker, Joseph Myers,
gcc Patches, kees Cook, isanbard
On Thu, Oct 26, 2023 at 7:22 AM Jakub Jelinek <jakub@redhat.com> wrote:
>
> On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
> > For the code generation impact:
> >
> > turning the original x.buf
> > to a builtin function call
> > __builtin_with_access_and_size(x,buf, x.L,-1)
> >
> > might inhibit some optimizations from happening before the builtin is
> > evaluated into object size info (phase .objsz1). I guess there might be
> > some performance impact.
> >
> > However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
> > performance impact will be reduced to minimum?
>
> You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
> be able to figure out the dynamic sizes in case of normal (non-early)
> inlining - caller takes address of a counted_by array, passes it down
> to callee which is only inlined late and uses __bdos, or callee takes address
> and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
>
> And while the builtin (or if it is an internal detail rather than user
> accessible builtin an internal function) could be even const/nothrow/leaf if
> the arguments contain the loads from the structure 2 fields, I'm afraid it
> will still have huge code generation impact, prevent tons of pre-IPA
> optimizations. And it will need some work to handle it properly during
> inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
> values, so it wouldn't be just the builtin/internal-fn call to be ignored,
> but also the count load from memory.
I think we want to track the value, not the "memory" in the builtin call,
so GIMPLE would be
_1 = x.L;
.. = __builtin_with_access_and_size (&x.buf, _1, -1);
also please make sure to use an internal function for
__builtin_with_access_and_size,
I don't think we want to expose this to users - it's an implementation detail.
Richard.
>
> Jakub
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 8:45 ` Richard Biener
@ 2023-10-26 9:20 ` Martin Uecker
2023-10-26 10:14 ` Martin Uecker
2023-10-26 16:41 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 9:20 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
> > >
> > > > Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
> > > >
> > > > Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> > > > > > On 2023-10-25 04:16, Martin Uecker wrote:
> > > > > > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > > > > > >
> > > > > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > > > >
> > > > > > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > > > > > Hi, Sid,
> > > > > > > > >
> > > > > > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > > > > > >
> > > > > > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > > > > > >
> > > > > > > > > 1 #include <malloc.h>
> > > > > > > > > 2 struct A
> > > > > > > > > 3 {
> > > > > > > > > 4 size_t size;
> > > > > > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > > > > > 6 };
> > > > > > > > > 7
> > > > > > > > > 8 static size_t
> > > > > > > > > 9 get_size_from (void *ptr)
> > > > > > > > > 10 {
> > > > > > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > > > > > 12 }
> > > > > > > > > 13
> > > > > > > > > 14 void
> > > > > > > > > 15 foo (size_t sz)
> > > > > > > > > 16 {
> > > > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > > > 18 obj->size = sz;
> > > > > > > > > 19 obj->buf[0] = 2;
> > > > > > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > > > > > 21 return;
> > > > > > > > > 22 }
> > > > > > > > > 23
> > > > > > > > > 24 int main ()
> > > > > > > > > 25 {
> > > > > > > > > 26 foo (20);
> > > > > > > > > 27 return 0;
> > > > > > > > > 28 }
> > > > > > > > >
> > > > >
> > > > > <snip>
> > > > >
> > > > > > > When it’s set I suppose. Turn
> > > > > > >
> > > > > > > X.l = n;
> > > > > > >
> > > > > > > Into
> > > > > > >
> > > > > > > X.l = __builtin_with_size (x.buf, n);
> > > > > >
> > > > > > It would turn
> > > > > >
> > > > > > some_variable = (&) x.buf
> > > > > >
> > > > > > into
> > > > > >
> > > > > > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> > > > > >
> > > > > >
> > > > > > So the later access to x.buf and not the initialization
> > > > > > of a member of the struct (which is too early).
> > > > > >
> > > > >
> > > > > Hmm, so with Qing's example above, are you suggesting the transformation
> > > > > be to foo like so:
> > > > >
> > > > > 14 void
> > > > > 15 foo (size_t sz)
> > > > > 16 {
> > > > > 16.5 void * _1;
> > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > 18 obj->size = sz;
> > > > > 19 obj->buf[0] = 2;
> > > > > 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> > > > > 20 __builtin_printf (“%d\n", get_size_from (_1));
> > > > > 21 return;
> > > > > 22 }
> > > > >
> > > > > If yes then this could indeed work. I think I got thrown off by the
> > > > > reference to __bdos.
> > > >
> > > > Yes. I think it is important not to evaluate the size at the
> > > > access to buf and not the allocation, because the point is to
> > > > recover it from the size member even when the compiler can't
> > > > see the original allocation.
> > >
> > > But if the access is through a pointer without the attribute visible
> > > even the Frontend cannot recover?
> >
> > Yes, if the access is using a struct-with-FAM without the attribute
> > the FE would not be insert the builtin. BDOS could potentially
> > still see the original allocation but if it doesn't, then there is
> > no information.
> >
> > > We’d need to force type correctness and give up on indirecting
> > > through an int * when it can refer to two diffenent container types.
> > > The best we can do I think is mark allocation sites and hope for
> > > some basic code hygiene (not clobbering size or array pointer
> > > through pointers without the appropriately attributed type)
> >
> > I am do not fully understand what you are referring to.
>
> struct A { int n; int data[n]; };
> struct B { long n; int data[n]; };
>
> int *p = flag ? a->data : b->data;
>
> access *p;
>
> Since we need to allow interoperability of pointers (a->data is
> convertible to a non-fat pointer of type int *) this leaves us with
> ambiguity we need to conservatively handle to avoid false positives.
For BDOS, I would expect this to work exactly like:
char aa[n1];
char bb[n2];
char *p = flag ? aa : bb;
(or similar code with malloc). In fact it does:
https://godbolt.org/z/bK68YKqhe
(cheating a bit and also the sub-object version of
BDOS does not seem to work)
>
> We _might_ want to diagnose decay of a->data to int *, but IIRC
> there's no way (or proposal) to allow declaring a corresponding
> fat pointer, so it's not a good designed feature.
As a language feature, I fully agree. I see the
counted_by attribute has a makeshift solution.
But we can already do:
auto p = flag ? &aa : &bb;
and this already works perfectly:
https://godbolt.org/z/rvb6xWWPj
We can also name the variably-modifed type:
char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
https://godbolt.org/z/13cTT1vGP
The problem with this version is that consistency
is not checked. (I have patch for adding run-time
checks).
And then the next step would be to allow
char (*p)[:] = flag ? &aa : &bb;
or similar. Dennis Ritchie proposed this himself
a long time ago.
So far this seems straightfoward.
If we then want to allow such wide pointers as
function arguments or in structs, we would need
to define an ABI. But the ABI could just be
struct { char (*p)[.s]; size_t s; };
Maybe we could try to make the following
ABI compatible:
int foo(int p[s], size_t s);
int foo(int p[:]);
> Having __builtin_with_size at allocation would possibly make
> the BOS use-def walk discover both objects.
Yes. But I do not think this there is any fundamental
difference to discovering allocation functions.
> I think you can't
> insert __builtin_with_size at the access to *p, but in practice
> that would be very much needed.
Usually the access to *p would follow directly the
access x.buf, so BDOS should find it.
But yes, to get full bounds safety, the pointer type
has to change to a variably-modified type (which would work
today) or a fat pointer type. The later can be built on
vm-types easily because all the FE semantics already
exists.
Martin
>
> Richard.
>
> > But yes,
> > for full bounds safety we would need the language feature.
> > In C people should start to variably-modified types
> > more. I think we can build perfect bounds safety on top of
> > them in a very good way with only FE changes.
> >
> > All these attributes are just a best effort. But for a while,
> > this will be necessary.
> >
> > Martin
> >
> > >
> > > > Evaluating at this point requires that the size is correctly set
> > > > before the access to the FAM and the user has to make sure
> > > > this is the case. But to me this requirement would make sense.
> > > >
> > > > Semantically, it could aöso make sense to evaluate the size at a
> > > > later time. But then the reordering becomes problematic again.
> > > >
> > > > Also I think this would make this feature generally more useful.
> > > > For example, it could work also for others pointers in the struct
> > > > and not just for FAMs. In this case, the struct may already be
> > > > freed when BDOS is called, so it might also not possible to
> > > > access the size member at a later time.
> > > >
> > > > Martin
> > > >
> > > >
> > > > >
> > > >
> >
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 9:20 ` Martin Uecker
@ 2023-10-26 10:14 ` Martin Uecker
2023-10-26 14:05 ` Richard Biener
2023-10-26 16:41 ` Qing Zhao
1 sibling, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 10:14 UTC (permalink / raw)
To: Richard Biener
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Donnerstag, dem 26.10.2023 um 11:20 +0200 schrieb Martin Uecker:
> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
> > On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
> > >
> > > Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
> > > >
> > > > > Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > >
> > > > > Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> > > > > > > On 2023-10-25 04:16, Martin Uecker wrote:
> > > > > > > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > > > > > > >
> > > > > > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > > > > >
> > > > > > > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > > > > > > Hi, Sid,
> > > > > > > > > >
> > > > > > > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > > > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > > > > > > >
> > > > > > > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > > > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > > > > > > >
> > > > > > > > > > 1 #include <malloc.h>
> > > > > > > > > > 2 struct A
> > > > > > > > > > 3 {
> > > > > > > > > > 4 size_t size;
> > > > > > > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > > > > > > 6 };
> > > > > > > > > > 7
> > > > > > > > > > 8 static size_t
> > > > > > > > > > 9 get_size_from (void *ptr)
> > > > > > > > > > 10 {
> > > > > > > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > > > > > > 12 }
> > > > > > > > > > 13
> > > > > > > > > > 14 void
> > > > > > > > > > 15 foo (size_t sz)
> > > > > > > > > > 16 {
> > > > > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > > > > 18 obj->size = sz;
> > > > > > > > > > 19 obj->buf[0] = 2;
> > > > > > > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > > > > > > 21 return;
> > > > > > > > > > 22 }
> > > > > > > > > > 23
> > > > > > > > > > 24 int main ()
> > > > > > > > > > 25 {
> > > > > > > > > > 26 foo (20);
> > > > > > > > > > 27 return 0;
> > > > > > > > > > 28 }
> > > > > > > > > >
> > > > > >
> > > > > > <snip>
> > > > > >
> > > > > > > > When it’s set I suppose. Turn
> > > > > > > >
> > > > > > > > X.l = n;
> > > > > > > >
> > > > > > > > Into
> > > > > > > >
> > > > > > > > X.l = __builtin_with_size (x.buf, n);
> > > > > > >
> > > > > > > It would turn
> > > > > > >
> > > > > > > some_variable = (&) x.buf
> > > > > > >
> > > > > > > into
> > > > > > >
> > > > > > > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> > > > > > >
> > > > > > >
> > > > > > > So the later access to x.buf and not the initialization
> > > > > > > of a member of the struct (which is too early).
> > > > > > >
> > > > > >
> > > > > > Hmm, so with Qing's example above, are you suggesting the transformation
> > > > > > be to foo like so:
> > > > > >
> > > > > > 14 void
> > > > > > 15 foo (size_t sz)
> > > > > > 16 {
> > > > > > 16.5 void * _1;
> > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > 18 obj->size = sz;
> > > > > > 19 obj->buf[0] = 2;
> > > > > > 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> > > > > > 20 __builtin_printf (“%d\n", get_size_from (_1));
> > > > > > 21 return;
> > > > > > 22 }
> > > > > >
> > > > > > If yes then this could indeed work. I think I got thrown off by the
> > > > > > reference to __bdos.
> > > > >
> > > > > Yes. I think it is important not to evaluate the size at the
> > > > > access to buf and not the allocation, because the point is to
> > > > > recover it from the size member even when the compiler can't
> > > > > see the original allocation.
> > > >
> > > > But if the access is through a pointer without the attribute visible
> > > > even the Frontend cannot recover?
> > >
> > > Yes, if the access is using a struct-with-FAM without the attribute
> > > the FE would not be insert the builtin. BDOS could potentially
> > > still see the original allocation but if it doesn't, then there is
> > > no information.
> > >
> > > > We’d need to force type correctness and give up on indirecting
> > > > through an int * when it can refer to two diffenent container types.
> > > > The best we can do I think is mark allocation sites and hope for
> > > > some basic code hygiene (not clobbering size or array pointer
> > > > through pointers without the appropriately attributed type)
> > >
> > > I am do not fully understand what you are referring to.
> >
> > struct A { int n; int data[n]; };
> > struct B { long n; int data[n]; };
> >
> > int *p = flag ? a->data : b->data;
> >
> > access *p;
> >
> > Since we need to allow interoperability of pointers (a->data is
> > convertible to a non-fat pointer of type int *) this leaves us with
> > ambiguity we need to conservatively handle to avoid false positives.
>
> For BDOS, I would expect this to work exactly like:
>
> char aa[n1];
> char bb[n2];
> char *p = flag ? aa : bb;
>
> (or similar code with malloc). In fact it does:
>
> https://godbolt.org/z/bK68YKqhe
> (cheating a bit and also the sub-object version of
> BDOS does not seem to work)
>
> >
> > We _might_ want to diagnose decay of a->data to int *, but IIRC
> > there's no way (or proposal) to allow declaring a corresponding
> > fat pointer, so it's not a good designed feature.
>
> As a language feature, I fully agree. I see the
> counted_by attribute has a makeshift solution.
>
> But we can already do:
>
> auto p = flag ? &aa : &bb;
>
> and this already works perfectly:
>
> https://godbolt.org/z/rvb6xWWPj
>
> We can also name the variably-modifed type:
>
> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
> https://godbolt.org/z/13cTT1vGP
>
> The problem with this version is that consistency
> is not checked. (I have patch for adding run-time
> checks).
>
> And then the next step would be to allow
>
> char (*p)[:] = flag ? &aa : &bb;
>
> or similar. Dennis Ritchie proposed this himself
> a long time ago.
>
> So far this seems straightfoward.
>
> If we then want to allow such wide pointers as
> function arguments or in structs, we would need
> to define an ABI. But the ABI could just be
>
> struct { char (*p)[.s]; size_t s; };
>
> Maybe we could try to make the following
> ABI compatible:
>
> int foo(int p[s], size_t s);
> int foo(int p[:]);
>
>
> > Having __builtin_with_size at allocation would possibly make
> > the BOS use-def walk discover both objects.
>
> Yes. But I do not think this there is any fundamental
> difference to discovering allocation functions.
>
> > I think you can't
> > insert __builtin_with_size at the access to *p, but in practice
> > that would be very much needed.
>
> Usually the access to *p would follow directly the
> access x.buf, so BDOS should find it.
>
> But yes, to get full bounds safety, the pointer type
> has to change to a variably-modified type (which would work
> today) or a fat pointer type. The later can be built on
> vm-types easily because all the FE semantics already
> exists.
We could insert the __builtin_with_size everywhere
we have to convert a wide pointer or let an array
decay to traditional pointer for reason of compatibility
with legacy code.
Martin
>
> Martin
>
> >
> > Richard.
> >
> > > But yes,
> > > for full bounds safety we would need the language feature.
> > > In C people should start to variably-modified types
> > > more. I think we can build perfect bounds safety on top of
> > > them in a very good way with only FE changes.
> > >
> > > All these attributes are just a best effort. But for a while,
> > > this will be necessary.
> > >
> > > Martin
> > >
> > > >
> > > > > Evaluating at this point requires that the size is correctly set
> > > > > before the access to the FAM and the user has to make sure
> > > > > this is the case. But to me this requirement would make sense.
> > > > >
> > > > > Semantically, it could aöso make sense to evaluate the size at a
> > > > > later time. But then the reordering becomes problematic again.
> > > > >
> > > > > Also I think this would make this feature generally more useful.
> > > > > For example, it could work also for others pointers in the struct
> > > > > and not just for FAMs. In this case, the struct may already be
> > > > > freed when BDOS is called, so it might also not possible to
> > > > > access the size member at a later time.
> > > > >
> > > > > Martin
> > > > >
> > > > >
> > > > > >
> > > > >
> > >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 10:14 ` Martin Uecker
@ 2023-10-26 14:05 ` Richard Biener
2023-10-26 18:54 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-26 14:05 UTC (permalink / raw)
To: Martin Uecker
Cc: Siddhesh Poyarekar, Qing Zhao, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 26.10.2023 um 12:14 schrieb Martin Uecker <uecker@tugraz.at>:
>
> Am Donnerstag, dem 26.10.2023 um 11:20 +0200 schrieb Martin Uecker:
>>> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
>>> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>>>>
>>>> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>>>>>
>>>>>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>
>>>>>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>>>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>>>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>>>>>
>>>>>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>>>>
>>>>>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>>>>>> Hi, Sid,
>>>>>>>>>>>
>>>>>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>>>>>
>>>>>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>>>>>
>>>>>>>>>>> 1 #include <malloc.h>
>>>>>>>>>>> 2 struct A
>>>>>>>>>>> 3 {
>>>>>>>>>>> 4 size_t size;
>>>>>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>>>>>> 6 };
>>>>>>>>>>> 7
>>>>>>>>>>> 8 static size_t
>>>>>>>>>>> 9 get_size_from (void *ptr)
>>>>>>>>>>> 10 {
>>>>>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>>>>>> 12 }
>>>>>>>>>>> 13
>>>>>>>>>>> 14 void
>>>>>>>>>>> 15 foo (size_t sz)
>>>>>>>>>>> 16 {
>>>>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>>>>> 18 obj->size = sz;
>>>>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>>>>>> 21 return;
>>>>>>>>>>> 22 }
>>>>>>>>>>> 23
>>>>>>>>>>> 24 int main ()
>>>>>>>>>>> 25 {
>>>>>>>>>>> 26 foo (20);
>>>>>>>>>>> 27 return 0;
>>>>>>>>>>> 28 }
>>>>>>>>>>>
>>>>>>>
>>>>>>> <snip>
>>>>>>>
>>>>>>>>> When it’s set I suppose. Turn
>>>>>>>>>
>>>>>>>>> X.l = n;
>>>>>>>>>
>>>>>>>>> Into
>>>>>>>>>
>>>>>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>>>>>
>>>>>>>> It would turn
>>>>>>>>
>>>>>>>> some_variable = (&) x.buf
>>>>>>>>
>>>>>>>> into
>>>>>>>>
>>>>>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>>>>>
>>>>>>>>
>>>>>>>> So the later access to x.buf and not the initialization
>>>>>>>> of a member of the struct (which is too early).
>>>>>>>>
>>>>>>>
>>>>>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>>>>>> be to foo like so:
>>>>>>>
>>>>>>> 14 void
>>>>>>> 15 foo (size_t sz)
>>>>>>> 16 {
>>>>>>> 16.5 void * _1;
>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>> 18 obj->size = sz;
>>>>>>> 19 obj->buf[0] = 2;
>>>>>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>>>>>> 21 return;
>>>>>>> 22 }
>>>>>>>
>>>>>>> If yes then this could indeed work. I think I got thrown off by the
>>>>>>> reference to __bdos.
>>>>>>
>>>>>> Yes. I think it is important not to evaluate the size at the
>>>>>> access to buf and not the allocation, because the point is to
>>>>>> recover it from the size member even when the compiler can't
>>>>>> see the original allocation.
>>>>>
>>>>> But if the access is through a pointer without the attribute visible
>>>>> even the Frontend cannot recover?
>>>>
>>>> Yes, if the access is using a struct-with-FAM without the attribute
>>>> the FE would not be insert the builtin. BDOS could potentially
>>>> still see the original allocation but if it doesn't, then there is
>>>> no information.
>>>>
>>>>> We’d need to force type correctness and give up on indirecting
>>>>> through an int * when it can refer to two diffenent container types.
>>>>> The best we can do I think is mark allocation sites and hope for
>>>>> some basic code hygiene (not clobbering size or array pointer
>>>>> through pointers without the appropriately attributed type)
>>>>
>>>> I am do not fully understand what you are referring to.
>>>
>>> struct A { int n; int data[n]; };
>>> struct B { long n; int data[n]; };
>>>
>>> int *p = flag ? a->data : b->data;
>>>
>>> access *p;
>>>
>>> Since we need to allow interoperability of pointers (a->data is
>>> convertible to a non-fat pointer of type int *) this leaves us with
>>> ambiguity we need to conservatively handle to avoid false positives.
>>
>> For BDOS, I would expect this to work exactly like:
>>
>> char aa[n1];
>> char bb[n2];
>> char *p = flag ? aa : bb;
>>
>> (or similar code with malloc). In fact it does:
>>
>> https://godbolt.org/z/bK68YKqhe
>> (cheating a bit and also the sub-object version of
>> BDOS does not seem to work)
>>
>>>
>>> We _might_ want to diagnose decay of a->data to int *, but IIRC
>>> there's no way (or proposal) to allow declaring a corresponding
>>> fat pointer, so it's not a good designed feature.
>>
>> As a language feature, I fully agree. I see the
>> counted_by attribute has a makeshift solution.
>>
>> But we can already do:
>>
>> auto p = flag ? &aa : &bb;
>>
>> and this already works perfectly:
>>
>> https://godbolt.org/z/rvb6xWWPj
>>
>> We can also name the variably-modifed type:
>>
>> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
>> https://godbolt.org/z/13cTT1vGP
>>
>> The problem with this version is that consistency
>> is not checked. (I have patch for adding run-time
>> checks).
>>
>> And then the next step would be to allow
>>
>> char (*p)[:] = flag ? &aa : &bb;
>>
>> or similar. Dennis Ritchie proposed this himself
>> a long time ago.
>>
>> So far this seems straightfoward.
>>
>> If we then want to allow such wide pointers as
>> function arguments or in structs, we would need
>> to define an ABI. But the ABI could just be
>>
>> struct { char (*p)[.s]; size_t s; };
>>
>> Maybe we could try to make the following
>> ABI compatible:
>>
>> int foo(int p[s], size_t s);
>> int foo(int p[:]);
>>
>>
>>> Having __builtin_with_size at allocation would possibly make
>>> the BOS use-def walk discover both objects.
>>
>> Yes. But I do not think this there is any fundamental
>> difference to discovering allocation functions.
>>
>>> I think you can't
>>> insert __builtin_with_size at the access to *p, but in practice
>>> that would be very much needed.
>>
>> Usually the access to *p would follow directly the
>> access x.buf, so BDOS should find it.
>>
>> But yes, to get full bounds safety, the pointer type
>> has to change to a variably-modified type (which would work
>> today) or a fat pointer type. The later can be built on
>> vm-types easily because all the FE semantics already
>> exists.
>
> We could insert the __builtin_with_size everywhere
> we have to convert a wide pointer or let an array
> decay to traditional pointer for reason of compatibility
> with legacy code.
That sounds like a nice idea. Note I’d like to see the consumer side implemented so we can play with different points of insertion (and I’ll try to show corner cases where it goes wrong). It all seems a bit late for GCC 14 though.
Richard
> Martin
>
>>
>> Martin
>>
>>>
>>> Richard.
>>>
>>>> But yes,
>>>> for full bounds safety we would need the language feature.
>>>> In C people should start to variably-modified types
>>>> more. I think we can build perfect bounds safety on top of
>>>> them in a very good way with only FE changes.
>>>>
>>>> All these attributes are just a best effort. But for a while,
>>>> this will be necessary.
>>>>
>>>> Martin
>>>>
>>>>>
>>>>>> Evaluating at this point requires that the size is correctly set
>>>>>> before the access to the FAM and the user has to make sure
>>>>>> this is the case. But to me this requirement would make sense.
>>>>>>
>>>>>> Semantically, it could aöso make sense to evaluate the size at a
>>>>>> later time. But then the reordering becomes problematic again.
>>>>>>
>>>>>> Also I think this would make this feature generally more useful.
>>>>>> For example, it could work also for others pointers in the struct
>>>>>> and not just for FAMs. In this case, the struct may already be
>>>>>> freed when BDOS is called, so it might also not possible to
>>>>>> access the size member at a later time.
>>>>>>
>>>>>> Martin
>>>>>>
>>>>>>
>>>>>>>
>>>>>>
>>>>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 5:21 ` Jakub Jelinek
2023-10-26 8:56 ` Richard Biener
@ 2023-10-26 14:41 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 14:41 UTC (permalink / raw)
To: Jakub Jelinek, Richard Biener
Cc: Siddhesh Poyarekar, Martin Uecker, Joseph Myers, gcc Patches,
kees Cook, isanbard
> On Oct 26, 2023, at 1:21 AM, Jakub Jelinek <jakub@redhat.com> wrote:
>
> On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
>> For the code generation impact:
>>
>> turning the original x.buf
>> to a builtin function call
>> __builtin_with_access_and_size(x,buf, x.L,-1)
>>
>> might inhibit some optimizations from happening before the builtin is
>> evaluated into object size info (phase .objsz1). I guess there might be
>> some performance impact.
>>
>> However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
>> performance impact will be reduced to minimum?
>
> You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
> be able to figure out the dynamic sizes in case of normal (non-early)
> inlining - caller takes address of a counted_by array, passes it down
> to callee which is only inlined late and uses __bdos, or callee takes address
> and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
I guess that I didn’t say it very clear previously. Let me explain again:
My understanding is, there are “early_objsz” phase and then later “objsz1” phase for -O[1|2|3].
For -Og, there are “early_objsz” and then later “objsz2”.
So, the “objsz1” I mentioned (for the case -O[1|2|3]) should be the same as the “objsz2” you mentioned above? -:)
It’s the second objsz phase.
In the second objsz phase, I believe that all the inlining (including early inlining and IPA inlining) are all applied?
>
> And while the builtin (or if it is an internal detail rather than user
> accessible builtin an internal function)
Okay, will use an “internal function” instead of “ builtin function”.
> could be even const/nothrow/leaf if
> the arguments contain the loads from the structure 2 fields, I'm afraid it
> will still have huge code generation impact, prevent tons of pre-IPA
> optimizations. And it will need some work to handle it properly during
> inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
> values, so it wouldn't be just the builtin/internal-fn call to be ignored,
> but also the count load from memory.
Are you worrying about the potential additional LOADs will change the inlining decision
since the inlining heuristic depends on the # of loads from memory?
In additional to the # of loads, the # of instructions and the # of calls of the function
might be increased too, will these have impact on inlining decision?
In addition to inlining decision, any other impact to other IPA optimizations?
thanks.
Qing
>
> Jakub
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 8:56 ` Richard Biener
@ 2023-10-26 14:58 ` Qing Zhao
2023-10-26 15:48 ` Richard Biener
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 14:58 UTC (permalink / raw)
To: Richard Biener, Jakub Jelinek
Cc: Siddhesh Poyarekar, Martin Uecker, Joseph Myers, gcc Patches,
kees Cook, isanbard
> On Oct 26, 2023, at 4:56 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
> On Thu, Oct 26, 2023 at 7:22 AM Jakub Jelinek <jakub@redhat.com> wrote:
>>
>> On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
>>> For the code generation impact:
>>>
>>> turning the original x.buf
>>> to a builtin function call
>>> __builtin_with_access_and_size(x,buf, x.L,-1)
>>>
>>> might inhibit some optimizations from happening before the builtin is
>>> evaluated into object size info (phase .objsz1). I guess there might be
>>> some performance impact.
>>>
>>> However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
>>> performance impact will be reduced to minimum?
>>
>> You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
>> be able to figure out the dynamic sizes in case of normal (non-early)
>> inlining - caller takes address of a counted_by array, passes it down
>> to callee which is only inlined late and uses __bdos, or callee takes address
>> and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
>>
>> And while the builtin (or if it is an internal detail rather than user
>> accessible builtin an internal function) could be even const/nothrow/leaf if
>> the arguments contain the loads from the structure 2 fields, I'm afraid it
>> will still have huge code generation impact, prevent tons of pre-IPA
>> optimizations. And it will need some work to handle it properly during
>> inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
>> values, so it wouldn't be just the builtin/internal-fn call to be ignored,
>> but also the count load from memory.
>
> I think we want to track the value, not the "memory" in the builtin call,
> so GIMPLE would be
>
> _1 = x.L;
> .. = __builtin_with_access_and_size (&x.buf, _1, -1);
Before adding the __builtin_with_access_and_size, the code is:
&x.buf
After inserting the built-in, it becomes:
_1 = x.L;
__builtin_with_access_and_size (&x.buf, _1, -1).
So, the # of total instructions, the # of LOADs, and the # of calls will all be increased.
There will be impact to the inlining decision definitely.
>
> also please make sure to use an internal function for
> __builtin_with_access_and_size,
> I don't think we want to expose this to users - it's an implementation detail.
Okay, will define it as an internal function (add it to internal-fn.def). -:)
Qing
>
> Richard.
>
>>
>> Jakub
>>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 14:58 ` Qing Zhao
@ 2023-10-26 15:48 ` Richard Biener
2023-10-26 16:16 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Richard Biener @ 2023-10-26 15:48 UTC (permalink / raw)
To: Qing Zhao
Cc: Jakub Jelinek, Siddhesh Poyarekar, Martin Uecker, Joseph Myers,
gcc Patches, kees Cook, isanbard
> Am 26.10.2023 um 16:58 schrieb Qing Zhao <qing.zhao@oracle.com>:
>
>
>
>> On Oct 26, 2023, at 4:56 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>>
>>> On Thu, Oct 26, 2023 at 7:22 AM Jakub Jelinek <jakub@redhat.com> wrote:
>>>
>>> On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
>>>> For the code generation impact:
>>>>
>>>> turning the original x.buf
>>>> to a builtin function call
>>>> __builtin_with_access_and_size(x,buf, x.L,-1)
>>>>
>>>> might inhibit some optimizations from happening before the builtin is
>>>> evaluated into object size info (phase .objsz1). I guess there might be
>>>> some performance impact.
>>>>
>>>> However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
>>>> performance impact will be reduced to minimum?
>>>
>>> You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
>>> be able to figure out the dynamic sizes in case of normal (non-early)
>>> inlining - caller takes address of a counted_by array, passes it down
>>> to callee which is only inlined late and uses __bdos, or callee takes address
>>> and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
>>>
>>> And while the builtin (or if it is an internal detail rather than user
>>> accessible builtin an internal function) could be even const/nothrow/leaf if
>>> the arguments contain the loads from the structure 2 fields, I'm afraid it
>>> will still have huge code generation impact, prevent tons of pre-IPA
>>> optimizations. And it will need some work to handle it properly during
>>> inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
>>> values, so it wouldn't be just the builtin/internal-fn call to be ignored,
>>> but also the count load from memory.
>>
>> I think we want to track the value, not the "memory" in the builtin call,
>> so GIMPLE would be
>>
>> _1 = x.L;
>> .. = __builtin_with_access_and_size (&x.buf, _1, -1);
>
> Before adding the __builtin_with_access_and_size, the code is:
>
> &x.buf
>
> After inserting the built-in, it becomes:
>
> _1 = x.L;
> __builtin_with_access_and_size (&x.buf, _1, -1).
>
>
> So, the # of total instructions, the # of LOADs, and the # of calls will all be increased.
> There will be impact to the inlining decision definitely.
Note we have to make sure, if x is a pointer and we want to instrument &x->buf that we
Can dereference x. Possibly doing
_1 = x ? x->Len : -1;
I’m not sure the C standard makes accessing x->Len unconditionally not undefined behavior when &x->buf is computed. Definitely it’s a violation of the abstract machine of Len is volatile qualified (but we can reject such counted_by or instantiations as volatile qualified types).
Richard
>
>>
>> also please make sure to use an internal function for
>> __builtin_with_access_and_size,
>> I don't think we want to expose this to users - it's an implementation detail.
>
> Okay, will define it as an internal function (add it to internal-fn.def). -:)
>
> Qing
>>
>> Richard.
>>
>>>
>>> Jakub
>>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 8:15 ` Martin Uecker
@ 2023-10-26 16:13 ` Kees Cook
2023-10-26 16:45 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Kees Cook @ 2023-10-26 16:13 UTC (permalink / raw)
To: Martin Uecker
Cc: Qing Zhao, Siddhesh Poyarekar, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
On Thu, Oct 26, 2023 at 10:15:10AM +0200, Martin Uecker wrote:
> but not this:
>
> char *p = &x->buf;
> x->count = 1;
> p[10] = 1; // !
This seems fine to me -- it's how I'd expect it to work: "10" is beyond
"1".
> (because the pointer is passed around the
> store to the counter)
>
> and also here the second store is then irrelevant
> for the access:
>
> x->count = 10;
> char* p = &x->buf;
> ...
> x->count = 1; // somewhere else
> ----
> p[9] = 1; // ok, because count matter when buf was accesssed.
This is less great, but I can understand why it happens. "p" loses the
association with "x". It'd be nice if "p" had to way to retain that it
was just an alias for x->buf, so future p access would check count.
But this appears to be an existing limitation in other areas where an
assignment will cause the loss of object association. (I've run into
this before.) It's just more surprising in the above example because in
the past the loss of association would cause __bdos() to revert back to
"SIZE_MAX" results ("I don't know the size") rather than an "outdated"
size, which may get us into unexpected places...
> IMHO this makes sense also from the user side and
> are the desirable semantics we discussed before.
>
> But can you take a look at this?
>
>
> This should simulate it fairly well:
> https://godbolt.org/z/xq89aM7Gr
>
> (the call to the noinline function would go away,
> but not necessarily its impact on optimization)
Yeah, this example should be a very rare situation: a leaf function is
changing the characteristics of the struct but returning a buffer within
it to the caller. The more likely glitch would be from:
int main()
{
struct foo *f = foo_alloc(7);
char *p = FAM_ACCESS(f, size, buf);
printf("%ld\n", __builtin_dynamic_object_size(p, 0));
test1(f); // or just "f->count = 10;" no function call needed
printf("%ld\n", __builtin_dynamic_object_size(p, 0));
return 0;
}
which reports:
7
7
instead of:
7
10
This kind of "get an alias" situation is pretty common in the kernel
as a way to have a convenient "handle" to the array. In the case of a
"fill the array without knowing the actual final size" code pattern,
things would immediately break:
struct foo *f;
char *p;
int i;
f = alloc(maximum_possible);
f->count = 0;
p = f->buf;
for (i; data_is_available() && i < maximum_possible; i++) {
f->count ++;
p[i] = next_data_item();
}
Now perhaps the problem here is that "count" cannot be used for a count
of "logically valid members in the array" but must always be a count of
"allocated member space in the array", which I guess is tolerable, but
isn't ideal -- I'd like to catch logic bugs in addition to allocation
bugs, but the latter is certainly much more important to catch.
--
Kees Cook
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 15:48 ` Richard Biener
@ 2023-10-26 16:16 ` Martin Uecker
0 siblings, 0 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 16:16 UTC (permalink / raw)
To: Richard Biener, Qing Zhao
Cc: Jakub Jelinek, Siddhesh Poyarekar, Joseph Myers, gcc Patches,
kees Cook, isanbard
Am Donnerstag, dem 26.10.2023 um 17:48 +0200 schrieb Richard Biener:
>
> > Am 26.10.2023 um 16:58 schrieb Qing Zhao <qing.zhao@oracle.com>:
> >
> >
> >
> > > On Oct 26, 2023, at 4:56 AM, Richard Biener <richard.guenther@gmail.com> wrote:
> > >
> > > > On Thu, Oct 26, 2023 at 7:22 AM Jakub Jelinek <jakub@redhat.com> wrote:
> > > >
> > > > On Wed, Oct 25, 2023 at 07:03:43PM +0000, Qing Zhao wrote:
> > > > > For the code generation impact:
> > > > >
> > > > > turning the original x.buf
> > > > > to a builtin function call
> > > > > __builtin_with_access_and_size(x,buf, x.L,-1)
> > > > >
> > > > > might inhibit some optimizations from happening before the builtin is
> > > > > evaluated into object size info (phase .objsz1). I guess there might be
> > > > > some performance impact.
> > > > >
> > > > > However, if we mark this builtin as PURE, NOTRROW, etc, then the negative
> > > > > performance impact will be reduced to minimum?
> > > >
> > > > You can't drop it during objsz1 pass though, otherwise __bdos wouldn't
> > > > be able to figure out the dynamic sizes in case of normal (non-early)
> > > > inlining - caller takes address of a counted_by array, passes it down
> > > > to callee which is only inlined late and uses __bdos, or callee takes address
> > > > and returns it and caller uses __bdos, etc. - so it would need to be objsz2.
> > > >
> > > > And while the builtin (or if it is an internal detail rather than user
> > > > accessible builtin an internal function) could be even const/nothrow/leaf if
> > > > the arguments contain the loads from the structure 2 fields, I'm afraid it
> > > > will still have huge code generation impact, prevent tons of pre-IPA
> > > > optimizations. And it will need some work to handle it properly during
> > > > inlining heuristics, because in GIMPLE the COMPONENT_REF loads aren't gimple
> > > > values, so it wouldn't be just the builtin/internal-fn call to be ignored,
> > > > but also the count load from memory.
> > >
> > > I think we want to track the value, not the "memory" in the builtin call,
> > > so GIMPLE would be
> > >
> > > _1 = x.L;
> > > .. = __builtin_with_access_and_size (&x.buf, _1, -1);
> >
> > Before adding the __builtin_with_access_and_size, the code is:
> >
> > &x.buf
> >
> > After inserting the built-in, it becomes:
> >
> > _1 = x.L;
> > __builtin_with_access_and_size (&x.buf, _1, -1).
> >
> >
> > So, the # of total instructions, the # of LOADs, and the # of calls will all be increased.
> > There will be impact to the inlining decision definitely.
>
> Note we have to make sure, if x is a pointer and we want to instrument &x->buf that we
> Can dereference x. Possibly doing
>
> _1 = x ? x->Len : -1;
>
> I’m not sure the C standard makes accessing x->Len unconditionally not undefined behavior when &x->buf is computed. Definitely it’s a violation of the abstract machine of Len is volatile qualified (but we can reject such counted_by or instantiations as volatile qualified types).
I believe it is implicit UB to do &x->buf if there is
no object *x because the wording assumes the existence
of an object. In that case accessing x->L should
be fine too.
In practice the access may trap for other reasons
(mprotect etc.), but I guess this is acceptable,
but should probably be documented...
We might need the x? to not run into trouble with
those offsetof implementations written using null
pointer. Although in this case maybe one could
hope that the load will get optimized anyway ...
Martin
>
> Richard
>
> >
> > >
> > > also please make sure to use an internal function for
> > > __builtin_with_access_and_size,
> > > I don't think we want to expose this to users - it's an implementation detail.
> >
> > Okay, will define it as an internal function (add it to internal-fn.def). -:)
> >
> > Qing
> > >
> > > Richard.
> > >
> > > >
> > > > Jakub
> > > >
> >
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 9:20 ` Martin Uecker
2023-10-26 10:14 ` Martin Uecker
@ 2023-10-26 16:41 ` Qing Zhao
2023-10-26 17:05 ` Martin Uecker
1 sibling, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 16:41 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 26, 2023, at 5:20 AM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
>> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>>>
>>> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>>>>
>>>>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>
>>>>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>>>>
>>>>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>>>
>>>>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>>>>> Hi, Sid,
>>>>>>>>>>
>>>>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>>>>
>>>>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>>>>
>>>>>>>>>> 1 #include <malloc.h>
>>>>>>>>>> 2 struct A
>>>>>>>>>> 3 {
>>>>>>>>>> 4 size_t size;
>>>>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>>>>> 6 };
>>>>>>>>>> 7
>>>>>>>>>> 8 static size_t
>>>>>>>>>> 9 get_size_from (void *ptr)
>>>>>>>>>> 10 {
>>>>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>>>>> 12 }
>>>>>>>>>> 13
>>>>>>>>>> 14 void
>>>>>>>>>> 15 foo (size_t sz)
>>>>>>>>>> 16 {
>>>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>>>> 18 obj->size = sz;
>>>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>>>>> 21 return;
>>>>>>>>>> 22 }
>>>>>>>>>> 23
>>>>>>>>>> 24 int main ()
>>>>>>>>>> 25 {
>>>>>>>>>> 26 foo (20);
>>>>>>>>>> 27 return 0;
>>>>>>>>>> 28 }
>>>>>>>>>>
>>>>>>
>>>>>> <snip>
>>>>>>
>>>>>>>> When it’s set I suppose. Turn
>>>>>>>>
>>>>>>>> X.l = n;
>>>>>>>>
>>>>>>>> Into
>>>>>>>>
>>>>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>>>>
>>>>>>> It would turn
>>>>>>>
>>>>>>> some_variable = (&) x.buf
>>>>>>>
>>>>>>> into
>>>>>>>
>>>>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>>>>
>>>>>>>
>>>>>>> So the later access to x.buf and not the initialization
>>>>>>> of a member of the struct (which is too early).
>>>>>>>
>>>>>>
>>>>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>>>>> be to foo like so:
>>>>>>
>>>>>> 14 void
>>>>>> 15 foo (size_t sz)
>>>>>> 16 {
>>>>>> 16.5 void * _1;
>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>> 18 obj->size = sz;
>>>>>> 19 obj->buf[0] = 2;
>>>>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>>>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>>>>> 21 return;
>>>>>> 22 }
>>>>>>
>>>>>> If yes then this could indeed work. I think I got thrown off by the
>>>>>> reference to __bdos.
>>>>>
>>>>> Yes. I think it is important not to evaluate the size at the
>>>>> access to buf and not the allocation, because the point is to
>>>>> recover it from the size member even when the compiler can't
>>>>> see the original allocation.
>>>>
>>>> But if the access is through a pointer without the attribute visible
>>>> even the Frontend cannot recover?
>>>
>>> Yes, if the access is using a struct-with-FAM without the attribute
>>> the FE would not be insert the builtin. BDOS could potentially
>>> still see the original allocation but if it doesn't, then there is
>>> no information.
>>>
>>>> We’d need to force type correctness and give up on indirecting
>>>> through an int * when it can refer to two diffenent container types.
>>>> The best we can do I think is mark allocation sites and hope for
>>>> some basic code hygiene (not clobbering size or array pointer
>>>> through pointers without the appropriately attributed type)
>>>
>>> I am do not fully understand what you are referring to.
>>
>> struct A { int n; int data[n]; };
>> struct B { long n; int data[n]; };
>>
>> int *p = flag ? a->data : b->data;
>>
>> access *p;
>>
>> Since we need to allow interoperability of pointers (a->data is
>> convertible to a non-fat pointer of type int *) this leaves us with
>> ambiguity we need to conservatively handle to avoid false positives.
>
> For BDOS, I would expect this to work exactly like:
>
> char aa[n1];
> char bb[n2];
> char *p = flag ? aa : bb;
>
> (or similar code with malloc). In fact it does:
>
> https://godbolt.org/z/bK68YKqhe
> (cheating a bit and also the sub-object version of
> BDOS does not seem to work)
>
>>
>> We _might_ want to diagnose decay of a->data to int *, but IIRC
>> there's no way (or proposal) to allow declaring a corresponding
>> fat pointer, so it's not a good designed feature.
>
> As a language feature, I fully agree. I see the
> counted_by attribute has a makeshift solution.
The “counted_by” attribute is necessary at this moment since
it will be much easier to be adopted by the existing source code,
for example, the Linux Kernel.
Though I agree that embedding the bound information into TYPE
system should be the ultimate goal.
>
> But we can already do:
>
> auto p = flag ? &aa : &bb;
>
> and this already works perfectly:
>
> https://godbolt.org/z/rvb6xWWPj
>
> We can also name the variably-modifed type:
>
> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
> https://godbolt.org/z/13cTT1vGP
>
> The problem with this version is that consistency
> is not checked. (I have patch for adding run-time
> checks).
>
> And then the next step would be to allow
>
> char (*p)[:] = flag ? &aa : &bb;
>
> or similar. Dennis Ritchie proposed this himself
> a long time ago.
>
> So far this seems straightfoward.
>
> If we then want to allow such wide pointers as
> function arguments or in structs, we would need
> to define an ABI. But the ABI could just be
>
> struct { char (*p)[.s]; size_t s; };
>
> Maybe we could try to make the following
> ABI compatible:
>
> int foo(int p[s], size_t s);
> int foo(int p[:]);
>
>
>> Having __builtin_with_size at allocation would possibly make
>> the BOS use-def walk discover both objects.
>
> Yes. But I do not think this there is any fundamental
> difference to discovering allocation functions.
>
>> I think you can't
>> insert __builtin_with_size at the access to *p, but in practice
>> that would be very much needed.
>
> Usually the access to *p would follow directly the
> access x.buf, so BDOS should find it.
>
> But yes, to get full bounds safety, the pointer type
> has to change to a variably-modified type (which would work
> today) or a fat pointer type.
By variable-modified type, you mean the VLA?
There is one major difference between VLA and (FAM or Pointer array):
For VLA, the compiler is responsible for allocating the memory for it,
the size assignment and the memory allocation are both done by the
compiler at the same time and tied together.
But for FAM and pointer arrays, right now, users allocate the memory for them
In the source code, so, when we add the “counted_by” attribute, we need to
specify the additional requirement for the order of size assignment and memory
allocation into the source code, and specify this requirement in the user documentation.
Later, if we try to make the bound information of FAM/pointer array into TYPE
system, similar as the current VLA, should we also need to move the memory allocation
of the FAM/pointer arrays into compiler (similar as VLA too)?
> The later can be built on
> vm-types easily because all the FE semantics already
> exists.
Except the memory allocation part…
Do I miss anything here?
Qing
>
> Martin
>
>>
>> Richard.
>>
>>> But yes,
>>> for full bounds safety we would need the language feature.
>>> In C people should start to variably-modified types
>>> more. I think we can build perfect bounds safety on top of
>>> them in a very good way with only FE changes.
>>>
>>> All these attributes are just a best effort. But for a while,
>>> this will be necessary.
>>>
>>> Martin
>>>
>>>>
>>>>> Evaluating at this point requires that the size is correctly set
>>>>> before the access to the FAM and the user has to make sure
>>>>> this is the case. But to me this requirement would make sense.
>>>>>
>>>>> Semantically, it could aöso make sense to evaluate the size at a
>>>>> later time. But then the reordering becomes problematic again.
>>>>>
>>>>> Also I think this would make this feature generally more useful.
>>>>> For example, it could work also for others pointers in the struct
>>>>> and not just for FAMs. In this case, the struct may already be
>>>>> freed when BDOS is called, so it might also not possible to
>>>>> access the size member at a later time.
>>>>>
>>>>> Martin
>>>>>
>>>>>
>>>>>>
>>>>>
>>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 16:13 ` Kees Cook
@ 2023-10-26 16:45 ` Martin Uecker
2023-10-26 19:57 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 16:45 UTC (permalink / raw)
To: Kees Cook
Cc: Qing Zhao, Siddhesh Poyarekar, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
Am Donnerstag, dem 26.10.2023 um 09:13 -0700 schrieb Kees Cook:
> On Thu, Oct 26, 2023 at 10:15:10AM +0200, Martin Uecker wrote:
> > but not this:
> >
x->count = 11;
> > char *p = &x->buf;
> > x->count = 1;
> > p[10] = 1; // !
>
> This seems fine to me -- it's how I'd expect it to work: "10" is beyond
> "1".
Note that the store would be allowed.
>
> > (because the pointer is passed around the
> > store to the counter)
> >
> > and also here the second store is then irrelevant
> > for the access:
> >
> > x->count = 10;
> > char* p = &x->buf;
> > ...
> > x->count = 1; // somewhere else
> > ----
> > p[9] = 1; // ok, because count matter when buf was accesssed.
>
> This is less great, but I can understand why it happens. "p" loses the
> association with "x". It'd be nice if "p" had to way to retain that it
> was just an alias for x->buf, so future p access would check count.
The problem is not to discover that p is an alias to x->buf,
but that it seems difficult to make sure that stores to
x->count are not reordered relative to the final access to
p[i] you want to check, so that you then get the right value.
>
> But this appears to be an existing limitation in other areas where an
> assignment will cause the loss of object association. (I've run into
> this before.) It's just more surprising in the above example because in
> the past the loss of association would cause __bdos() to revert back to
> "SIZE_MAX" results ("I don't know the size") rather than an "outdated"
> size, which may get us into unexpected places...
>
> > IMHO this makes sense also from the user side and
> > are the desirable semantics we discussed before.
> >
> > But can you take a look at this?
> >
> >
> > This should simulate it fairly well:
> > https://godbolt.org/z/xq89aM7Gr
> >
> > (the call to the noinline function would go away,
> > but not necessarily its impact on optimization)
>
> Yeah, this example should be a very rare situation: a leaf function is
> changing the characteristics of the struct but returning a buffer within
> it to the caller. The more likely glitch would be from:
>
> int main()
> {
> struct foo *f = foo_alloc(7);
> char *p = FAM_ACCESS(f, size, buf);
>
> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
> test1(f); // or just "f->count = 10;" no function call needed
> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
>
> return 0;
> }
>
> which reports:
> 7
> 7
>
> instead of:
> 7
> 10
>
> This kind of "get an alias" situation is pretty common in the kernel
> as a way to have a convenient "handle" to the array. In the case of a
> "fill the array without knowing the actual final size" code pattern,
> things would immediately break:
>
> struct foo *f;
> char *p;
> int i;
>
> f = alloc(maximum_possible);
> f->count = 0;
> p = f->buf;
>
> for (i; data_is_available() && i < maximum_possible; i++) {
> f->count ++;
> p[i] = next_data_item();
> }
>
> Now perhaps the problem here is that "count" cannot be used for a count
> of "logically valid members in the array" but must always be a count of
> "allocated member space in the array", which I guess is tolerable, but
> isn't ideal -- I'd like to catch logic bugs in addition to allocation
> bugs, but the latter is certainly much more important to catch.
Maybe we could have a warning when f->buf is not directly
accessed.
Martin
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 16:41 ` Qing Zhao
@ 2023-10-26 17:05 ` Martin Uecker
2023-10-26 17:35 ` Richard Biener
2023-10-26 19:20 ` Qing Zhao
0 siblings, 2 replies; 116+ messages in thread
From: Martin Uecker @ 2023-10-26 17:05 UTC (permalink / raw)
To: Qing Zhao
Cc: Richard Biener, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
Am Donnerstag, dem 26.10.2023 um 16:41 +0000 schrieb Qing Zhao:
>
> > On Oct 26, 2023, at 5:20 AM, Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
> > > On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
> > > >
> > > > Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
> > > > >
> > > > > > Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > >
> > > > > > Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
> > > > > > > > On 2023-10-25 04:16, Martin Uecker wrote:
> > > > > > > > Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
> > > > > > > > >
> > > > > > > > > > Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
> > > > > > > > > >
> > > > > > > > > > Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
> > > > > > > > > > > Hi, Sid,
> > > > > > > > > > >
> > > > > > > > > > > Really appreciate for your example and detailed explanation. Very helpful.
> > > > > > > > > > > I think that this example is an excellent example to show (almost) all the issues we need to consider.
> > > > > > > > > > >
> > > > > > > > > > > I slightly modified this example to make it to be compilable and run-able, as following:
> > > > > > > > > > > (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
> > > > > > > > > > >
> > > > > > > > > > > 1 #include <malloc.h>
> > > > > > > > > > > 2 struct A
> > > > > > > > > > > 3 {
> > > > > > > > > > > 4 size_t size;
> > > > > > > > > > > 5 char buf[] __attribute__((counted_by(size)));
> > > > > > > > > > > 6 };
> > > > > > > > > > > 7
> > > > > > > > > > > 8 static size_t
> > > > > > > > > > > 9 get_size_from (void *ptr)
> > > > > > > > > > > 10 {
> > > > > > > > > > > 11 return __builtin_dynamic_object_size (ptr, 1);
> > > > > > > > > > > 12 }
> > > > > > > > > > > 13
> > > > > > > > > > > 14 void
> > > > > > > > > > > 15 foo (size_t sz)
> > > > > > > > > > > 16 {
> > > > > > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > > > > > 18 obj->size = sz;
> > > > > > > > > > > 19 obj->buf[0] = 2;
> > > > > > > > > > > 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
> > > > > > > > > > > 21 return;
> > > > > > > > > > > 22 }
> > > > > > > > > > > 23
> > > > > > > > > > > 24 int main ()
> > > > > > > > > > > 25 {
> > > > > > > > > > > 26 foo (20);
> > > > > > > > > > > 27 return 0;
> > > > > > > > > > > 28 }
> > > > > > > > > > >
> > > > > > >
> > > > > > > <snip>
> > > > > > >
> > > > > > > > > When it’s set I suppose. Turn
> > > > > > > > >
> > > > > > > > > X.l = n;
> > > > > > > > >
> > > > > > > > > Into
> > > > > > > > >
> > > > > > > > > X.l = __builtin_with_size (x.buf, n);
> > > > > > > >
> > > > > > > > It would turn
> > > > > > > >
> > > > > > > > some_variable = (&) x.buf
> > > > > > > >
> > > > > > > > into
> > > > > > > >
> > > > > > > > some_variable = __builtin_with_size ( (&) x.buf. x.len)
> > > > > > > >
> > > > > > > >
> > > > > > > > So the later access to x.buf and not the initialization
> > > > > > > > of a member of the struct (which is too early).
> > > > > > > >
> > > > > > >
> > > > > > > Hmm, so with Qing's example above, are you suggesting the transformation
> > > > > > > be to foo like so:
> > > > > > >
> > > > > > > 14 void
> > > > > > > 15 foo (size_t sz)
> > > > > > > 16 {
> > > > > > > 16.5 void * _1;
> > > > > > > 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
> > > > > > > 18 obj->size = sz;
> > > > > > > 19 obj->buf[0] = 2;
> > > > > > > 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
> > > > > > > 20 __builtin_printf (“%d\n", get_size_from (_1));
> > > > > > > 21 return;
> > > > > > > 22 }
> > > > > > >
> > > > > > > If yes then this could indeed work. I think I got thrown off by the
> > > > > > > reference to __bdos.
> > > > > >
> > > > > > Yes. I think it is important not to evaluate the size at the
> > > > > > access to buf and not the allocation, because the point is to
> > > > > > recover it from the size member even when the compiler can't
> > > > > > see the original allocation.
> > > > >
> > > > > But if the access is through a pointer without the attribute visible
> > > > > even the Frontend cannot recover?
> > > >
> > > > Yes, if the access is using a struct-with-FAM without the attribute
> > > > the FE would not be insert the builtin. BDOS could potentially
> > > > still see the original allocation but if it doesn't, then there is
> > > > no information.
> > > >
> > > > > We’d need to force type correctness and give up on indirecting
> > > > > through an int * when it can refer to two diffenent container types.
> > > > > The best we can do I think is mark allocation sites and hope for
> > > > > some basic code hygiene (not clobbering size or array pointer
> > > > > through pointers without the appropriately attributed type)
> > > >
> > > > I am do not fully understand what you are referring to.
> > >
> > > struct A { int n; int data[n]; };
> > > struct B { long n; int data[n]; };
> > >
> > > int *p = flag ? a->data : b->data;
> > >
> > > access *p;
> > >
> > > Since we need to allow interoperability of pointers (a->data is
> > > convertible to a non-fat pointer of type int *) this leaves us with
> > > ambiguity we need to conservatively handle to avoid false positives.
> >
> > For BDOS, I would expect this to work exactly like:
> >
> > char aa[n1];
> > char bb[n2];
> > char *p = flag ? aa : bb;
> >
> > (or similar code with malloc). In fact it does:
> >
> > https://godbolt.org/z/bK68YKqhe
> > (cheating a bit and also the sub-object version of
> > BDOS does not seem to work)
> >
> > >
> > > We _might_ want to diagnose decay of a->data to int *, but IIRC
> > > there's no way (or proposal) to allow declaring a corresponding
> > > fat pointer, so it's not a good designed feature.
> >
> > As a language feature, I fully agree. I see the
> > counted_by attribute has a makeshift solution.
>
> The “counted_by” attribute is necessary at this moment since
> it will be much easier to be adopted by the existing source code,
> for example, the Linux Kernel.
Yes, this is understood.
>
> Though I agree that embedding the bound information into TYPE
> system should be the ultimate goal.
>
> >
> > But we can already do:
> >
> > auto p = flag ? &aa : &bb;
> >
> > and this already works perfectly:
> >
> > https://godbolt.org/z/rvb6xWWPj
> >
> > We can also name the variably-modifed type:
> >
> > char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
> > https://godbolt.org/z/13cTT1vGP
> >
> > The problem with this version is that consistency
> > is not checked. (I have patch for adding run-time
> > checks).
> >
> > And then the next step would be to allow
> >
> > char (*p)[:] = flag ? &aa : &bb;
> >
> > or similar. Dennis Ritchie proposed this himself
> > a long time ago.
> >
> > So far this seems straightfoward.
> >
> > If we then want to allow such wide pointers as
> > function arguments or in structs, we would need
> > to define an ABI. But the ABI could just be
> >
> > struct { char (*p)[.s]; size_t s; };
> >
> > Maybe we could try to make the following
> > ABI compatible:
> >
> > int foo(int p[s], size_t s);
> > int foo(int p[:]);
> >
> >
> > > Having __builtin_with_size at allocation would possibly make
> > > the BOS use-def walk discover both objects.
> >
> > Yes. But I do not think this there is any fundamental
> > difference to discovering allocation functions.
> >
> > > I think you can't
> > > insert __builtin_with_size at the access to *p, but in practice
> > > that would be very much needed.
> >
> > Usually the access to *p would follow directly the
> > access x.buf, so BDOS should find it.
> >
> > But yes, to get full bounds safety, the pointer type
> > has to change to a variably-modified type (which would work
> > today) or a fat pointer type.
>
> By variable-modified type, you mean the VLA?
I mean a pointer to a VLA type.
>
> There is one major difference between VLA and (FAM or Pointer array):
>
> For VLA, the compiler is responsible for allocating the memory for it,
> the size assignment and the memory allocation are both done by the
> compiler at the same time and tied together.
A VLA can also exist on the heap:
char (*buf)[n] = malloc(sizeof(*buf));
>
> But for FAM and pointer arrays, right now, users allocate the memory for them
> In the source code, so, when we add the “counted_by” attribute, we need to
> specify the additional requirement for the order of size assignment and memory
> allocation into the source code, and specify this requirement in the user documentation.
>
> Later, if we try to make the bound information of FAM/pointer array into TYPE
> system, similar as the current VLA, should we also need to move the memory allocation
> of the FAM/pointer arrays into compiler (similar as VLA too)?
I think memory allocation can be done either
as an automatic variable or by malloc.
The following works today in GNU C:
int N = ..;
struct foo { char buf[N]; } x;
struct foo *p = malloc(sizeof(struct foo));
The only limitation today is that the size 'n'
can not refer to the field member.
struct foo { int n; char buf[.n]; };
I am not yet sure how we would set the size for
an automatic object, but I have some ideas. Maybe
simply using an initializer:
struct foo x = { .n = 10 };
Martin
> > The later can be built on
> > vm-types easily because all the FE semantics already
> > exists.
>
> Except the memory allocation part…
>
> Do I miss anything here?
>
> Qing
> >
> > Martin
> >
> > >
> > > Richard.
> > >
> > > > But yes,
> > > > for full bounds safety we would need the language feature.
> > > > In C people should start to variably-modified types
> > > > more. I think we can build perfect bounds safety on top of
> > > > them in a very good way with only FE changes.
> > > >
> > > > All these attributes are just a best effort. But for a while,
> > > > this will be necessary.
> > > >
> > > > Martin
> > > >
> > > > >
> > > > > > Evaluating at this point requires that the size is correctly set
> > > > > > before the access to the FAM and the user has to make sure
> > > > > > this is the case. But to me this requirement would make sense.
> > > > > >
> > > > > > Semantically, it could aöso make sense to evaluate the size at a
> > > > > > later time. But then the reordering becomes problematic again.
> > > > > >
> > > > > > Also I think this would make this feature generally more useful.
> > > > > > For example, it could work also for others pointers in the struct
> > > > > > and not just for FAMs. In this case, the struct may already be
> > > > > > freed when BDOS is called, so it might also not possible to
> > > > > > access the size member at a later time.
> > > > > >
> > > > > > Martin
> > > > > >
> > > > > >
> > > > > > >
> > > > > >
> > > >
> >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 17:05 ` Martin Uecker
@ 2023-10-26 17:35 ` Richard Biener
2023-10-26 19:20 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Richard Biener @ 2023-10-26 17:35 UTC (permalink / raw)
To: Martin Uecker
Cc: Qing Zhao, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> Am 26.10.2023 um 19:05 schrieb Martin Uecker <uecker@tugraz.at>:
>
> Am Donnerstag, dem 26.10.2023 um 16:41 +0000 schrieb Qing Zhao:
>>
>>>> On Oct 26, 2023, at 5:20 AM, Martin Uecker <uecker@tugraz.at> wrote:
>>>
>>> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
>>>> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>>>>>
>>>>> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>>>>>>
>>>>>>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>
>>>>>>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>>>>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>>>>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>>>>>>
>>>>>>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>>>>>
>>>>>>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>>>>>>> Hi, Sid,
>>>>>>>>>>>>
>>>>>>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>>>>>>
>>>>>>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>>>>>>
>>>>>>>>>>>> 1 #include <malloc.h>
>>>>>>>>>>>> 2 struct A
>>>>>>>>>>>> 3 {
>>>>>>>>>>>> 4 size_t size;
>>>>>>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>>>>>>> 6 };
>>>>>>>>>>>> 7
>>>>>>>>>>>> 8 static size_t
>>>>>>>>>>>> 9 get_size_from (void *ptr)
>>>>>>>>>>>> 10 {
>>>>>>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>>>>>>> 12 }
>>>>>>>>>>>> 13
>>>>>>>>>>>> 14 void
>>>>>>>>>>>> 15 foo (size_t sz)
>>>>>>>>>>>> 16 {
>>>>>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>>>>>> 18 obj->size = sz;
>>>>>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>>>>>>> 21 return;
>>>>>>>>>>>> 22 }
>>>>>>>>>>>> 23
>>>>>>>>>>>> 24 int main ()
>>>>>>>>>>>> 25 {
>>>>>>>>>>>> 26 foo (20);
>>>>>>>>>>>> 27 return 0;
>>>>>>>>>>>> 28 }
>>>>>>>>>>>>
>>>>>>>>
>>>>>>>> <snip>
>>>>>>>>
>>>>>>>>>> When it’s set I suppose. Turn
>>>>>>>>>>
>>>>>>>>>> X.l = n;
>>>>>>>>>>
>>>>>>>>>> Into
>>>>>>>>>>
>>>>>>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>>>>>>
>>>>>>>>> It would turn
>>>>>>>>>
>>>>>>>>> some_variable = (&) x.buf
>>>>>>>>>
>>>>>>>>> into
>>>>>>>>>
>>>>>>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>>>>>>
>>>>>>>>>
>>>>>>>>> So the later access to x.buf and not the initialization
>>>>>>>>> of a member of the struct (which is too early).
>>>>>>>>>
>>>>>>>>
>>>>>>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>>>>>>> be to foo like so:
>>>>>>>>
>>>>>>>> 14 void
>>>>>>>> 15 foo (size_t sz)
>>>>>>>> 16 {
>>>>>>>> 16.5 void * _1;
>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>> 18 obj->size = sz;
>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>>>>>>> 21 return;
>>>>>>>> 22 }
>>>>>>>>
>>>>>>>> If yes then this could indeed work. I think I got thrown off by the
>>>>>>>> reference to __bdos.
>>>>>>>
>>>>>>> Yes. I think it is important not to evaluate the size at the
>>>>>>> access to buf and not the allocation, because the point is to
>>>>>>> recover it from the size member even when the compiler can't
>>>>>>> see the original allocation.
>>>>>>
>>>>>> But if the access is through a pointer without the attribute visible
>>>>>> even the Frontend cannot recover?
>>>>>
>>>>> Yes, if the access is using a struct-with-FAM without the attribute
>>>>> the FE would not be insert the builtin. BDOS could potentially
>>>>> still see the original allocation but if it doesn't, then there is
>>>>> no information.
>>>>>
>>>>>> We’d need to force type correctness and give up on indirecting
>>>>>> through an int * when it can refer to two diffenent container types.
>>>>>> The best we can do I think is mark allocation sites and hope for
>>>>>> some basic code hygiene (not clobbering size or array pointer
>>>>>> through pointers without the appropriately attributed type)
>>>>>
>>>>> I am do not fully understand what you are referring to.
>>>>
>>>> struct A { int n; int data[n]; };
>>>> struct B { long n; int data[n]; };
>>>>
>>>> int *p = flag ? a->data : b->data;
>>>>
>>>> access *p;
>>>>
>>>> Since we need to allow interoperability of pointers (a->data is
>>>> convertible to a non-fat pointer of type int *) this leaves us with
>>>> ambiguity we need to conservatively handle to avoid false positives.
>>>
>>> For BDOS, I would expect this to work exactly like:
>>>
>>> char aa[n1];
>>> char bb[n2];
>>> char *p = flag ? aa : bb;
>>>
>>> (or similar code with malloc). In fact it does:
>>>
>>> https://godbolt.org/z/bK68YKqhe
>>> (cheating a bit and also the sub-object version of
>>> BDOS does not seem to work)
>>>
>>>>
>>>> We _might_ want to diagnose decay of a->data to int *, but IIRC
>>>> there's no way (or proposal) to allow declaring a corresponding
>>>> fat pointer, so it's not a good designed feature.
>>>
>>> As a language feature, I fully agree. I see the
>>> counted_by attribute has a makeshift solution.
>>
>> The “counted_by” attribute is necessary at this moment since
>> it will be much easier to be adopted by the existing source code,
>> for example, the Linux Kernel.
>
> Yes, this is understood.
>
>>
>> Though I agree that embedding the bound information into TYPE
>> system should be the ultimate goal.
>>
>>>
>>> But we can already do:
>>>
>>> auto p = flag ? &aa : &bb;
>>>
>>> and this already works perfectly:
>>>
>>> https://godbolt.org/z/rvb6xWWPj
>>>
>>> We can also name the variably-modifed type:
>>>
>>> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
>>> https://godbolt.org/z/13cTT1vGP
>>>
>>> The problem with this version is that consistency
>>> is not checked. (I have patch for adding run-time
>>> checks).
>>>
>>> And then the next step would be to allow
>>>
>>> char (*p)[:] = flag ? &aa : &bb;
>>>
>>> or similar. Dennis Ritchie proposed this himself
>>> a long time ago.
>>>
>>> So far this seems straightfoward.
>>>
>>> If we then want to allow such wide pointers as
>>> function arguments or in structs, we would need
>>> to define an ABI. But the ABI could just be
>>>
>>> struct { char (*p)[.s]; size_t s; };
>>>
>>> Maybe we could try to make the following
>>> ABI compatible:
>>>
>>> int foo(int p[s], size_t s);
>>> int foo(int p[:]);
>>>
>>>
>>>> Having __builtin_with_size at allocation would possibly make
>>>> the BOS use-def walk discover both objects.
>>>
>>> Yes. But I do not think this there is any fundamental
>>> difference to discovering allocation functions.
>>>
>>>> I think you can't
>>>> insert __builtin_with_size at the access to *p, but in practice
>>>> that would be very much needed.
>>>
>>> Usually the access to *p would follow directly the
>>> access x.buf, so BDOS should find it.
>>>
>>> But yes, to get full bounds safety, the pointer type
>>> has to change to a variably-modified type (which would work
>>> today) or a fat pointer type.
>>
>> By variable-modified type, you mean the VLA?
>
> I mean a pointer to a VLA type.
>
>>
>> There is one major difference between VLA and (FAM or Pointer array):
>>
>> For VLA, the compiler is responsible for allocating the memory for it,
>> the size assignment and the memory allocation are both done by the
>> compiler at the same time and tied together.
>
> A VLA can also exist on the heap:
>
> char (*buf)[n] = malloc(sizeof(*buf));
>
>>
>> But for FAM and pointer arrays, right now, users allocate the memory for them
>> In the source code, so, when we add the “counted_by” attribute, we need to
>> specify the additional requirement for the order of size assignment and memory
>> allocation into the source code, and specify this requirement in the user documentation.
>>
>> Later, if we try to make the bound information of FAM/pointer array into TYPE
>> system, similar as the current VLA, should we also need to move the memory allocation
>> of the FAM/pointer arrays into compiler (similar as VLA too)?
>
> I think memory allocation can be done either
> as an automatic variable or by malloc.
>
> The following works today in GNU C:
>
> int N = ..;
> struct foo { char buf[N]; } x;
> struct foo *p = malloc(sizeof(struct foo));
>
> The only limitation today is that the size 'n'
> can not refer to the field member.
>
> struct foo { int n; char buf[.n]; };
Note the middle end is perfectly capable of
This since at least Ada extensively supports this kind of type layout. You just need to extend the parser to accept it (and properly declare the FIELD_DECLS). Of course it’s only half of the story since int[n] still decays to int * losing the information again.
Richard
> I am not yet sure how we would set the size for
> an automatic object, but I have some ideas. Maybe
> simply using an initializer:
>
> struct foo x = { .n = 10 };
>
>
> Martin
>
>>> The later can be built on
>>> vm-types easily because all the FE semantics already
>>> exists.
>>
>> Except the memory allocation part…
>>
>> Do I miss anything here?
>>
>> Qing
>>>
>>> Martin
>>>
>>>>
>>>> Richard.
>>>>
>>>>> But yes,
>>>>> for full bounds safety we would need the language feature.
>>>>> In C people should start to variably-modified types
>>>>> more. I think we can build perfect bounds safety on top of
>>>>> them in a very good way with only FE changes.
>>>>>
>>>>> All these attributes are just a best effort. But for a while,
>>>>> this will be necessary.
>>>>>
>>>>> Martin
>>>>>
>>>>>>
>>>>>>> Evaluating at this point requires that the size is correctly set
>>>>>>> before the access to the FAM and the user has to make sure
>>>>>>> this is the case. But to me this requirement would make sense.
>>>>>>>
>>>>>>> Semantically, it could aöso make sense to evaluate the size at a
>>>>>>> later time. But then the reordering becomes problematic again.
>>>>>>>
>>>>>>> Also I think this would make this feature generally more useful.
>>>>>>> For example, it could work also for others pointers in the struct
>>>>>>> and not just for FAMs. In this case, the struct may already be
>>>>>>> freed when BDOS is called, so it might also not possible to
>>>>>>> access the size member at a later time.
>>>>>>>
>>>>>>> Martin
>>>>>>>
>>>>>>>
>>>>>>>>
>>>>>>>
>>>>>
>>>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 14:05 ` Richard Biener
@ 2023-10-26 18:54 ` Qing Zhao
2023-10-27 16:43 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 18:54 UTC (permalink / raw)
To: Richard Biener, Martin uecker
Cc: Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek, gcc Patches,
kees Cook, isanbard
> On Oct 26, 2023, at 10:05 AM, Richard Biener <richard.guenther@gmail.com> wrote:
>
>
>
>> Am 26.10.2023 um 12:14 schrieb Martin Uecker <uecker@tugraz.at>:
>>
>> Am Donnerstag, dem 26.10.2023 um 11:20 +0200 schrieb Martin Uecker:
>>>> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
>>>> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>>>>>
>>>>> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>>>>>>
>>>>>>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>
>>>>>>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>>>>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>>>>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>>>>>>
>>>>>>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>>>>>
>>>>>>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>>>>>>> Hi, Sid,
>>>>>>>>>>>>
>>>>>>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>>>>>>
>>>>>>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>>>>>>
>>>>>>>>>>>> 1 #include <malloc.h>
>>>>>>>>>>>> 2 struct A
>>>>>>>>>>>> 3 {
>>>>>>>>>>>> 4 size_t size;
>>>>>>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>>>>>>> 6 };
>>>>>>>>>>>> 7
>>>>>>>>>>>> 8 static size_t
>>>>>>>>>>>> 9 get_size_from (void *ptr)
>>>>>>>>>>>> 10 {
>>>>>>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>>>>>>> 12 }
>>>>>>>>>>>> 13
>>>>>>>>>>>> 14 void
>>>>>>>>>>>> 15 foo (size_t sz)
>>>>>>>>>>>> 16 {
>>>>>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>>>>>> 18 obj->size = sz;
>>>>>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>>>>>>> 21 return;
>>>>>>>>>>>> 22 }
>>>>>>>>>>>> 23
>>>>>>>>>>>> 24 int main ()
>>>>>>>>>>>> 25 {
>>>>>>>>>>>> 26 foo (20);
>>>>>>>>>>>> 27 return 0;
>>>>>>>>>>>> 28 }
>>>>>>>>>>>>
>>>>>>>>
>>>>>>>> <snip>
>>>>>>>>
>>>>>>>>>> When it’s set I suppose. Turn
>>>>>>>>>>
>>>>>>>>>> X.l = n;
>>>>>>>>>>
>>>>>>>>>> Into
>>>>>>>>>>
>>>>>>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>>>>>>
>>>>>>>>> It would turn
>>>>>>>>>
>>>>>>>>> some_variable = (&) x.buf
>>>>>>>>>
>>>>>>>>> into
>>>>>>>>>
>>>>>>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>>>>>>
>>>>>>>>>
>>>>>>>>> So the later access to x.buf and not the initialization
>>>>>>>>> of a member of the struct (which is too early).
>>>>>>>>>
>>>>>>>>
>>>>>>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>>>>>>> be to foo like so:
>>>>>>>>
>>>>>>>> 14 void
>>>>>>>> 15 foo (size_t sz)
>>>>>>>> 16 {
>>>>>>>> 16.5 void * _1;
>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>> 18 obj->size = sz;
>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>>>>>>> 21 return;
>>>>>>>> 22 }
>>>>>>>>
>>>>>>>> If yes then this could indeed work. I think I got thrown off by the
>>>>>>>> reference to __bdos.
>>>>>>>
>>>>>>> Yes. I think it is important not to evaluate the size at the
>>>>>>> access to buf and not the allocation, because the point is to
>>>>>>> recover it from the size member even when the compiler can't
>>>>>>> see the original allocation.
>>>>>>
>>>>>> But if the access is through a pointer without the attribute visible
>>>>>> even the Frontend cannot recover?
>>>>>
>>>>> Yes, if the access is using a struct-with-FAM without the attribute
>>>>> the FE would not be insert the builtin. BDOS could potentially
>>>>> still see the original allocation but if it doesn't, then there is
>>>>> no information.
>>>>>
>>>>>> We’d need to force type correctness and give up on indirecting
>>>>>> through an int * when it can refer to two diffenent container types.
>>>>>> The best we can do I think is mark allocation sites and hope for
>>>>>> some basic code hygiene (not clobbering size or array pointer
>>>>>> through pointers without the appropriately attributed type)
>>>>>
>>>>> I am do not fully understand what you are referring to.
>>>>
>>>> struct A { int n; int data[n]; };
>>>> struct B { long n; int data[n]; };
>>>>
>>>> int *p = flag ? a->data : b->data;
>>>>
>>>> access *p;
>>>>
>>>> Since we need to allow interoperability of pointers (a->data is
>>>> convertible to a non-fat pointer of type int *) this leaves us with
>>>> ambiguity we need to conservatively handle to avoid false positives.
>>>
>>> For BDOS, I would expect this to work exactly like:
>>>
>>> char aa[n1];
>>> char bb[n2];
>>> char *p = flag ? aa : bb;
>>>
>>> (or similar code with malloc). In fact it does:
>>>
>>> https://godbolt.org/z/bK68YKqhe
>>> (cheating a bit and also the sub-object version of
>>> BDOS does not seem to work)
>>>
>>>>
>>>> We _might_ want to diagnose decay of a->data to int *, but IIRC
>>>> there's no way (or proposal) to allow declaring a corresponding
>>>> fat pointer, so it's not a good designed feature.
>>>
>>> As a language feature, I fully agree. I see the
>>> counted_by attribute has a makeshift solution.
>>>
>>> But we can already do:
>>>
>>> auto p = flag ? &aa : &bb;
>>>
>>> and this already works perfectly:
>>>
>>> https://godbolt.org/z/rvb6xWWPj
>>>
>>> We can also name the variably-modifed type:
>>>
>>> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
>>> https://godbolt.org/z/13cTT1vGP
>>>
>>> The problem with this version is that consistency
>>> is not checked. (I have patch for adding run-time
>>> checks).
>>>
>>> And then the next step would be to allow
>>>
>>> char (*p)[:] = flag ? &aa : &bb;
>>>
>>> or similar. Dennis Ritchie proposed this himself
>>> a long time ago.
>>>
>>> So far this seems straightfoward.
>>>
>>> If we then want to allow such wide pointers as
>>> function arguments or in structs, we would need
>>> to define an ABI. But the ABI could just be
>>>
>>> struct { char (*p)[.s]; size_t s; };
>>>
>>> Maybe we could try to make the following
>>> ABI compatible:
>>>
>>> int foo(int p[s], size_t s);
>>> int foo(int p[:]);
>>>
>>>
>>>> Having __builtin_with_size at allocation would possibly make
>>>> the BOS use-def walk discover both objects.
>>>
>>> Yes. But I do not think this there is any fundamental
>>> difference to discovering allocation functions.
>>>
>>>> I think you can't
>>>> insert __builtin_with_size at the access to *p, but in practice
>>>> that would be very much needed.
>>>
>>> Usually the access to *p would follow directly the
>>> access x.buf, so BDOS should find it.
>>>
>>> But yes, to get full bounds safety, the pointer type
>>> has to change to a variably-modified type (which would work
>>> today) or a fat pointer type. The later can be built on
>>> vm-types easily because all the FE semantics already
>>> exists.
>>
>> We could insert the __builtin_with_size everywhere
>> we have to convert a wide pointer or let an array
>> decay to traditional pointer for reason of compatibility
>> with legacy code.
>
> That sounds like a nice idea. Note I’d like to see the consumer side implemented so we can play with different points of insertion (and I’ll try to show corner cases where it goes wrong).
Giving the example I mentioned previously:
1 #include <malloc.h>
2 struct A
3 {
4 size_t size;
5 char buf[] __attribute__((counted_by(size)));
6 };
7
8 static size_t
9 get_size_from (void *ptr)
10 {
11 return __builtin_dynamic_object_size (ptr, 1);
12 }
13
14 void
15 foo (size_t sz)
16 {
17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
18 obj->size = sz;
19 obj->buf[0] = 2;
20 __builtin_printf (“%d\n", get_size_from (obj->buf));
21 return;
22 }
So, the different points of insertion the new __builtin_with_size in FE include the following points: (per my understanding so far)
Point 1. When the “obj->buf” is referenced at line 19, and line 20?
Point 2. When the “obj” is allocated at line 17?
Are these correct?
Any other points we need to consider?
> It all seems a bit late for GCC 14 though.
Yes, I agree. Given the potential impact on the code generation and other potential issues, it’s better to be put in the early stage of the next release.
Qing
>
> Richard
>
>> Martin
>>
>>>
>>> Martin
>>>
>>>>
>>>> Richard.
>>>>
>>>>> But yes,
>>>>> for full bounds safety we would need the language feature.
>>>>> In C people should start to variably-modified types
>>>>> more. I think we can build perfect bounds safety on top of
>>>>> them in a very good way with only FE changes.
>>>>>
>>>>> All these attributes are just a best effort. But for a while,
>>>>> this will be necessary.
>>>>>
>>>>> Martin
>>>>>
>>>>>>
>>>>>>> Evaluating at this point requires that the size is correctly set
>>>>>>> before the access to the FAM and the user has to make sure
>>>>>>> this is the case. But to me this requirement would make sense.
>>>>>>>
>>>>>>> Semantically, it could aöso make sense to evaluate the size at a
>>>>>>> later time. But then the reordering becomes problematic again.
>>>>>>>
>>>>>>> Also I think this would make this feature generally more useful.
>>>>>>> For example, it could work also for others pointers in the struct
>>>>>>> and not just for FAMs. In this case, the struct may already be
>>>>>>> freed when BDOS is called, so it might also not possible to
>>>>>>> access the size member at a later time.
>>>>>>>
>>>>>>> Martin
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 17:05 ` Martin Uecker
2023-10-26 17:35 ` Richard Biener
@ 2023-10-26 19:20 ` Qing Zhao
1 sibling, 0 replies; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 19:20 UTC (permalink / raw)
To: Martin Uecker
Cc: Richard Biener, Siddhesh Poyarekar, Joseph Myers, Jakub Jelinek,
gcc Patches, kees Cook, isanbard
> On Oct 26, 2023, at 1:05 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Donnerstag, dem 26.10.2023 um 16:41 +0000 schrieb Qing Zhao:
>>
>>> On Oct 26, 2023, at 5:20 AM, Martin Uecker <uecker@tugraz.at> wrote:
>>>
>>> Am Donnerstag, dem 26.10.2023 um 10:45 +0200 schrieb Richard Biener:
>>>> On Wed, Oct 25, 2023 at 8:16 PM Martin Uecker <uecker@tugraz.at> wrote:
>>>>>
>>>>> Am Mittwoch, dem 25.10.2023 um 13:13 +0200 schrieb Richard Biener:
>>>>>>
>>>>>>> Am 25.10.2023 um 12:47 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>
>>>>>>> Am Mittwoch, dem 25.10.2023 um 06:25 -0400 schrieb Siddhesh Poyarekar:
>>>>>>>>> On 2023-10-25 04:16, Martin Uecker wrote:
>>>>>>>>> Am Mittwoch, dem 25.10.2023 um 08:43 +0200 schrieb Richard Biener:
>>>>>>>>>>
>>>>>>>>>>> Am 24.10.2023 um 22:38 schrieb Martin Uecker <uecker@tugraz.at>:
>>>>>>>>>>>
>>>>>>>>>>> Am Dienstag, dem 24.10.2023 um 20:30 +0000 schrieb Qing Zhao:
>>>>>>>>>>>> Hi, Sid,
>>>>>>>>>>>>
>>>>>>>>>>>> Really appreciate for your example and detailed explanation. Very helpful.
>>>>>>>>>>>> I think that this example is an excellent example to show (almost) all the issues we need to consider.
>>>>>>>>>>>>
>>>>>>>>>>>> I slightly modified this example to make it to be compilable and run-able, as following:
>>>>>>>>>>>> (but I still cannot make the incorrect reordering or DSE happening, anyway, the potential reordering possibility is there…)
>>>>>>>>>>>>
>>>>>>>>>>>> 1 #include <malloc.h>
>>>>>>>>>>>> 2 struct A
>>>>>>>>>>>> 3 {
>>>>>>>>>>>> 4 size_t size;
>>>>>>>>>>>> 5 char buf[] __attribute__((counted_by(size)));
>>>>>>>>>>>> 6 };
>>>>>>>>>>>> 7
>>>>>>>>>>>> 8 static size_t
>>>>>>>>>>>> 9 get_size_from (void *ptr)
>>>>>>>>>>>> 10 {
>>>>>>>>>>>> 11 return __builtin_dynamic_object_size (ptr, 1);
>>>>>>>>>>>> 12 }
>>>>>>>>>>>> 13
>>>>>>>>>>>> 14 void
>>>>>>>>>>>> 15 foo (size_t sz)
>>>>>>>>>>>> 16 {
>>>>>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>>>>>> 18 obj->size = sz;
>>>>>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (obj->buf));
>>>>>>>>>>>> 21 return;
>>>>>>>>>>>> 22 }
>>>>>>>>>>>> 23
>>>>>>>>>>>> 24 int main ()
>>>>>>>>>>>> 25 {
>>>>>>>>>>>> 26 foo (20);
>>>>>>>>>>>> 27 return 0;
>>>>>>>>>>>> 28 }
>>>>>>>>>>>>
>>>>>>>>
>>>>>>>> <snip>
>>>>>>>>
>>>>>>>>>> When it’s set I suppose. Turn
>>>>>>>>>>
>>>>>>>>>> X.l = n;
>>>>>>>>>>
>>>>>>>>>> Into
>>>>>>>>>>
>>>>>>>>>> X.l = __builtin_with_size (x.buf, n);
>>>>>>>>>
>>>>>>>>> It would turn
>>>>>>>>>
>>>>>>>>> some_variable = (&) x.buf
>>>>>>>>>
>>>>>>>>> into
>>>>>>>>>
>>>>>>>>> some_variable = __builtin_with_size ( (&) x.buf. x.len)
>>>>>>>>>
>>>>>>>>>
>>>>>>>>> So the later access to x.buf and not the initialization
>>>>>>>>> of a member of the struct (which is too early).
>>>>>>>>>
>>>>>>>>
>>>>>>>> Hmm, so with Qing's example above, are you suggesting the transformation
>>>>>>>> be to foo like so:
>>>>>>>>
>>>>>>>> 14 void
>>>>>>>> 15 foo (size_t sz)
>>>>>>>> 16 {
>>>>>>>> 16.5 void * _1;
>>>>>>>> 17 struct A *obj = __builtin_malloc (sizeof(struct A) + sz * sizeof(char));
>>>>>>>> 18 obj->size = sz;
>>>>>>>> 19 obj->buf[0] = 2;
>>>>>>>> 19.5 _1 = __builtin_with_size (obj->buf, obj->size);
>>>>>>>> 20 __builtin_printf (“%d\n", get_size_from (_1));
>>>>>>>> 21 return;
>>>>>>>> 22 }
>>>>>>>>
>>>>>>>> If yes then this could indeed work. I think I got thrown off by the
>>>>>>>> reference to __bdos.
>>>>>>>
>>>>>>> Yes. I think it is important not to evaluate the size at the
>>>>>>> access to buf and not the allocation, because the point is to
>>>>>>> recover it from the size member even when the compiler can't
>>>>>>> see the original allocation.
>>>>>>
>>>>>> But if the access is through a pointer without the attribute visible
>>>>>> even the Frontend cannot recover?
>>>>>
>>>>> Yes, if the access is using a struct-with-FAM without the attribute
>>>>> the FE would not be insert the builtin. BDOS could potentially
>>>>> still see the original allocation but if it doesn't, then there is
>>>>> no information.
>>>>>
>>>>>> We’d need to force type correctness and give up on indirecting
>>>>>> through an int * when it can refer to two diffenent container types.
>>>>>> The best we can do I think is mark allocation sites and hope for
>>>>>> some basic code hygiene (not clobbering size or array pointer
>>>>>> through pointers without the appropriately attributed type)
>>>>>
>>>>> I am do not fully understand what you are referring to.
>>>>
>>>> struct A { int n; int data[n]; };
>>>> struct B { long n; int data[n]; };
>>>>
>>>> int *p = flag ? a->data : b->data;
>>>>
>>>> access *p;
>>>>
>>>> Since we need to allow interoperability of pointers (a->data is
>>>> convertible to a non-fat pointer of type int *) this leaves us with
>>>> ambiguity we need to conservatively handle to avoid false positives.
>>>
>>> For BDOS, I would expect this to work exactly like:
>>>
>>> char aa[n1];
>>> char bb[n2];
>>> char *p = flag ? aa : bb;
>>>
>>> (or similar code with malloc). In fact it does:
>>>
>>> https://godbolt.org/z/bK68YKqhe
>>> (cheating a bit and also the sub-object version of
>>> BDOS does not seem to work)
>>>
>>>>
>>>> We _might_ want to diagnose decay of a->data to int *, but IIRC
>>>> there's no way (or proposal) to allow declaring a corresponding
>>>> fat pointer, so it's not a good designed feature.
>>>
>>> As a language feature, I fully agree. I see the
>>> counted_by attribute has a makeshift solution.
>>
>> The “counted_by” attribute is necessary at this moment since
>> it will be much easier to be adopted by the existing source code,
>> for example, the Linux Kernel.
>
> Yes, this is understood.
>
>>
>> Though I agree that embedding the bound information into TYPE
>> system should be the ultimate goal.
>>
>>>
>>> But we can already do:
>>>
>>> auto p = flag ? &aa : &bb;
>>>
>>> and this already works perfectly:
>>>
>>> https://godbolt.org/z/rvb6xWWPj
>>>
>>> We can also name the variably-modifed type:
>>>
>>> char (*p)[flag ? n1 : n2] = flag ? &aa : &bb;
>>> https://godbolt.org/z/13cTT1vGP
>>>
>>> The problem with this version is that consistency
>>> is not checked. (I have patch for adding run-time
>>> checks).
>>>
>>> And then the next step would be to allow
>>>
>>> char (*p)[:] = flag ? &aa : &bb;
>>>
>>> or similar. Dennis Ritchie proposed this himself
>>> a long time ago.
>>>
>>> So far this seems straightfoward.
>>>
>>> If we then want to allow such wide pointers as
>>> function arguments or in structs, we would need
>>> to define an ABI. But the ABI could just be
>>>
>>> struct { char (*p)[.s]; size_t s; };
>>>
>>> Maybe we could try to make the following
>>> ABI compatible:
>>>
>>> int foo(int p[s], size_t s);
>>> int foo(int p[:]);
>>>
>>>
>>>> Having __builtin_with_size at allocation would possibly make
>>>> the BOS use-def walk discover both objects.
>>>
>>> Yes. But I do not think this there is any fundamental
>>> difference to discovering allocation functions.
>>>
>>>> I think you can't
>>>> insert __builtin_with_size at the access to *p, but in practice
>>>> that would be very much needed.
>>>
>>> Usually the access to *p would follow directly the
>>> access x.buf, so BDOS should find it.
>>>
>>> But yes, to get full bounds safety, the pointer type
>>> has to change to a variably-modified type (which would work
>>> today) or a fat pointer type.
>>
>> By variable-modified type, you mean the VLA?
>
> I mean a pointer to a VLA type.
>
>>
>> There is one major difference between VLA and (FAM or Pointer array):
>>
>> For VLA, the compiler is responsible for allocating the memory for it,
>> the size assignment and the memory allocation are both done by the
>> compiler at the same time and tied together.
>
> A VLA can also exist on the heap:
>
> char (*buf)[n] = malloc(sizeof(*buf));
Okay. I see.
>
>>
>> But for FAM and pointer arrays, right now, users allocate the memory for them
>> In the source code, so, when we add the “counted_by” attribute, we need to
>> specify the additional requirement for the order of size assignment and memory
>> allocation into the source code, and specify this requirement in the user documentation.
>>
>> Later, if we try to make the bound information of FAM/pointer array into TYPE
>> system, similar as the current VLA, should we also need to move the memory allocation
>> of the FAM/pointer arrays into compiler (similar as VLA too)?
>
> I think memory allocation can be done either
> as an automatic variable or by malloc.
>
> The following works today in GNU C:
>
> int N = ..;
> struct foo { char buf[N]; } x;
> struct foo *p = malloc(sizeof(struct foo));
Yes, tried this, did work. -:) thanks.
Qing
>
> The only limitation today is that the size 'n'
> can not refer to the field member.
>
> struct foo { int n; char buf[.n]; };
>
> I am not yet sure how we would set the size for
> an automatic object, but I have some ideas. Maybe
> simply using an initializer:
>
> struct foo x = { .n = 10 };
>
>
> Martin
>
>>> The later can be built on
>>> vm-types easily because all the FE semantics already
>>> exists.
>>
>> Except the memory allocation part…
>>
>> Do I miss anything here?
>>
>> Qing
>>>
>>> Martin
>>>
>>>>
>>>> Richard.
>>>>
>>>>> But yes,
>>>>> for full bounds safety we would need the language feature.
>>>>> In C people should start to variably-modified types
>>>>> more. I think we can build perfect bounds safety on top of
>>>>> them in a very good way with only FE changes.
>>>>>
>>>>> All these attributes are just a best effort. But for a while,
>>>>> this will be necessary.
>>>>>
>>>>> Martin
>>>>>
>>>>>>
>>>>>>> Evaluating at this point requires that the size is correctly set
>>>>>>> before the access to the FAM and the user has to make sure
>>>>>>> this is the case. But to me this requirement would make sense.
>>>>>>>
>>>>>>> Semantically, it could aöso make sense to evaluate the size at a
>>>>>>> later time. But then the reordering becomes problematic again.
>>>>>>>
>>>>>>> Also I think this would make this feature generally more useful.
>>>>>>> For example, it could work also for others pointers in the struct
>>>>>>> and not just for FAMs. In this case, the struct may already be
>>>>>>> freed when BDOS is called, so it might also not possible to
>>>>>>> access the size member at a later time.
>>>>>>>
>>>>>>> Martin
>>>>>>>
>>>>>>>
>>>>>>>>
>>>>>>>
>>>>>
>>>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 16:45 ` Martin Uecker
@ 2023-10-26 19:57 ` Qing Zhao
2023-10-27 7:21 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-26 19:57 UTC (permalink / raw)
To: Martin Uecker, Kees Cook
Cc: Kees Cook, Siddhesh Poyarekar, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
I guess that what Kees wanted, ""fill the array without knowing the actual final size" code pattern”, as following:
>> struct foo *f;
>> char *p;
>> int i;
>>
>> f = alloc(maximum_possible);
>> f->count = 0;
>> p = f->buf;
>>
>> for (i; data_is_available() && i < maximum_possible; i++) {
>> f->count ++;
>> p[i] = next_data_item();
>> }
actually is a dynamic array, or more accurately, Bounded-size dynamic array: ( but not a dynamic allocated array as we discussed so far)
https://en.wikipedia.org/wiki/Dynamic_array
This dynamic array, also is called growable array, or resizable array, whose size can
be changed during the lifetime.
For VLA or FAM, I believe that they are both dynamic allocated array, i.e, even though the size is not know at the compilation time, but the size
will be fixed after the array is allocated.
I am not sure whether C has support to such Dynamic array? Or whether it’s easy to provide dynamic array support in C?
Qing
> On Oct 26, 2023, at 12:45 PM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Donnerstag, dem 26.10.2023 um 09:13 -0700 schrieb Kees Cook:
>> On Thu, Oct 26, 2023 at 10:15:10AM +0200, Martin Uecker wrote:
>>> but not this:
>>>
>
> x->count = 11;
>>> char *p = &x->buf;
>>> x->count = 1;
>>> p[10] = 1; // !
>>
>> This seems fine to me -- it's how I'd expect it to work: "10" is beyond
>> "1".
>
> Note that the store would be allowed.
>
>>
>>> (because the pointer is passed around the
>>> store to the counter)
>>>
>>> and also here the second store is then irrelevant
>>> for the access:
>>>
>>> x->count = 10;
>>> char* p = &x->buf;
>>> ...
>>> x->count = 1; // somewhere else
>>> ----
>>> p[9] = 1; // ok, because count matter when buf was accesssed.
>>
>> This is less great, but I can understand why it happens. "p" loses the
>> association with "x". It'd be nice if "p" had to way to retain that it
>> was just an alias for x->buf, so future p access would check count.
>
> The problem is not to discover that p is an alias to x->buf,
> but that it seems difficult to make sure that stores to
> x->count are not reordered relative to the final access to
> p[i] you want to check, so that you then get the right value.
>
>>
>> But this appears to be an existing limitation in other areas where an
>> assignment will cause the loss of object association. (I've run into
>> this before.) It's just more surprising in the above example because in
>> the past the loss of association would cause __bdos() to revert back to
>> "SIZE_MAX" results ("I don't know the size") rather than an "outdated"
>> size, which may get us into unexpected places...
>>
>>> IMHO this makes sense also from the user side and
>>> are the desirable semantics we discussed before.
>>>
>>> But can you take a look at this?
>>>
>>>
>>> This should simulate it fairly well:
>>> https://godbolt.org/z/xq89aM7Gr
>>>
>>> (the call to the noinline function would go away,
>>> but not necessarily its impact on optimization)
>>
>> Yeah, this example should be a very rare situation: a leaf function is
>> changing the characteristics of the struct but returning a buffer within
>> it to the caller. The more likely glitch would be from:
>>
>> int main()
>> {
>> struct foo *f = foo_alloc(7);
>> char *p = FAM_ACCESS(f, size, buf);
>>
>> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
>> test1(f); // or just "f->count = 10;" no function call needed
>> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
>>
>> return 0;
>> }
>>
>> which reports:
>> 7
>> 7
>>
>> instead of:
>> 7
>> 10
>>
>> This kind of "get an alias" situation is pretty common in the kernel
>> as a way to have a convenient "handle" to the array. In the case of a
>> "fill the array without knowing the actual final size" code pattern,
>> things would immediately break:
>>
>> struct foo *f;
>> char *p;
>> int i;
>>
>> f = alloc(maximum_possible);
>> f->count = 0;
>> p = f->buf;
>>
>> for (i; data_is_available() && i < maximum_possible; i++) {
>> f->count ++;
>> p[i] = next_data_item();
>> }
>>
>> Now perhaps the problem here is that "count" cannot be used for a count
>> of "logically valid members in the array" but must always be a count of
>> "allocated member space in the array", which I guess is tolerable, but
>> isn't ideal -- I'd like to catch logic bugs in addition to allocation
>> bugs, but the latter is certainly much more important to catch.
>
> Maybe we could have a warning when f->buf is not directly
> accessed.
>
> Martin
>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-26 19:57 ` Qing Zhao
@ 2023-10-27 7:21 ` Martin Uecker
2023-10-27 14:32 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-27 7:21 UTC (permalink / raw)
To: Qing Zhao, Kees Cook
Cc: Siddhesh Poyarekar, Richard Biener, Joseph Myers, Jakub Jelinek,
gcc Patches, isanbard
Am Donnerstag, dem 26.10.2023 um 19:57 +0000 schrieb Qing Zhao:
> I guess that what Kees wanted, ""fill the array without knowing the actual final size" code pattern”, as following:
>
> > > struct foo *f;
> > > char *p;
> > > int i;
> > >
> > > f = alloc(maximum_possible);
> > > f->count = 0;
> > > p = f->buf;
> > >
> > > for (i; data_is_available() && i < maximum_possible; i++) {
> > > f->count ++;
> > > p[i] = next_data_item();
> > > }
>
> actually is a dynamic array, or more accurately, Bounded-size dynamic array: ( but not a dynamic allocated array as we discussed so far)
>
> https://en.wikipedia.org/wiki/Dynamic_array
>
> This dynamic array, also is called growable array, or resizable array, whose size can
> be changed during the lifetime.
>
> For VLA or FAM, I believe that they are both dynamic allocated array, i.e, even though the size is not know at the compilation time, but the size
> will be fixed after the array is allocated.
>
> I am not sure whether C has support to such Dynamic array? Or whether it’s easy to provide dynamic array support in C?
It is possible to support dynamic arrays in C even with
good checking, but not safely using the pattern above
where you derive a pointer which you later use independently.
While we could track the connection to the original struct,
the necessary synchronization between the counter and the
access to the buffer is difficult. I do not see how this
could be supported with reasonable effort and cost.
But with this restriction in mind, we can do a lot in C.
For example, see my experimental (!) container library
which has vector type.
https://github.com/uecker/noplate/blob/main/test.c
You can get an array view for the vector (which then
also can decay to a pointer), so it interoperates nicely
with C but you can get good bounds checking.
But once you derive a pointer and pass it on, it gets
difficult. But if you want safety, you just have to
to simply avoid this in code.
What we could potentially do is add restrictions so
that the access to buf always has to go via x->buf
or you get at least a warning.
Martin
>
> Qing
>
>
> > On Oct 26, 2023, at 12:45 PM, Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Donnerstag, dem 26.10.2023 um 09:13 -0700 schrieb Kees Cook:
> > > On Thu, Oct 26, 2023 at 10:15:10AM +0200, Martin Uecker wrote:
> > > > but not this:
> > > >
> >
> > x->count = 11;
> > > > char *p = &x->buf;
> > > > x->count = 1;
> > > > p[10] = 1; // !
> > >
> > > This seems fine to me -- it's how I'd expect it to work: "10" is beyond
> > > "1".
> >
> > Note that the store would be allowed.
> >
> > >
> > > > (because the pointer is passed around the
> > > > store to the counter)
> > > >
> > > > and also here the second store is then irrelevant
> > > > for the access:
> > > >
> > > > x->count = 10;
> > > > char* p = &x->buf;
> > > > ...
> > > > x->count = 1; // somewhere else
> > > > ----
> > > > p[9] = 1; // ok, because count matter when buf was accesssed.
> > >
> > > This is less great, but I can understand why it happens. "p" loses the
> > > association with "x". It'd be nice if "p" had to way to retain that it
> > > was just an alias for x->buf, so future p access would check count.
> >
> > The problem is not to discover that p is an alias to x->buf,
> > but that it seems difficult to make sure that stores to
> > x->count are not reordered relative to the final access to
> > p[i] you want to check, so that you then get the right value.
> >
> > >
> > > But this appears to be an existing limitation in other areas where an
> > > assignment will cause the loss of object association. (I've run into
> > > this before.) It's just more surprising in the above example because in
> > > the past the loss of association would cause __bdos() to revert back to
> > > "SIZE_MAX" results ("I don't know the size") rather than an "outdated"
> > > size, which may get us into unexpected places...
> > >
> > > > IMHO this makes sense also from the user side and
> > > > are the desirable semantics we discussed before.
> > > >
> > > > But can you take a look at this?
> > > >
> > > >
> > > > This should simulate it fairly well:
> > > > https://godbolt.org/z/xq89aM7Gr
> > > >
> > > > (the call to the noinline function would go away,
> > > > but not necessarily its impact on optimization)
> > >
> > > Yeah, this example should be a very rare situation: a leaf function is
> > > changing the characteristics of the struct but returning a buffer within
> > > it to the caller. The more likely glitch would be from:
> > >
> > > int main()
> > > {
> > > struct foo *f = foo_alloc(7);
> > > char *p = FAM_ACCESS(f, size, buf);
> > >
> > > printf("%ld\n", __builtin_dynamic_object_size(p, 0));
> > > test1(f); // or just "f->count = 10;" no function call needed
> > > printf("%ld\n", __builtin_dynamic_object_size(p, 0));
> > >
> > > return 0;
> > > }
> > >
> > > which reports:
> > > 7
> > > 7
> > >
> > > instead of:
> > > 7
> > > 10
> > >
> > > This kind of "get an alias" situation is pretty common in the kernel
> > > as a way to have a convenient "handle" to the array. In the case of a
> > > "fill the array without knowing the actual final size" code pattern,
> > > things would immediately break:
> > >
> > > struct foo *f;
> > > char *p;
> > > int i;
> > >
> > > f = alloc(maximum_possible);
> > > f->count = 0;
> > > p = f->buf;
> > >
> > > for (i; data_is_available() && i < maximum_possible; i++) {
> > > f->count ++;
> > > p[i] = next_data_item();
> > > }
> > >
> > > Now perhaps the problem here is that "count" cannot be used for a count
> > > of "logically valid members in the array" but must always be a count of
> > > "allocated member space in the array", which I guess is tolerable, but
> > > isn't ideal -- I'd like to catch logic bugs in addition to allocation
> > > bugs, but the latter is certainly much more important to catch.
> >
> > Maybe we could have a warning when f->buf is not directly
> > accessed.
> >
> > Martin
> >
> > >
> >
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-27 7:21 ` Martin Uecker
@ 2023-10-27 14:32 ` Qing Zhao
2023-10-27 14:53 ` Martin Uecker
0 siblings, 1 reply; 116+ messages in thread
From: Qing Zhao @ 2023-10-27 14:32 UTC (permalink / raw)
To: Martin Uecker
Cc: Kees Cook, Siddhesh Poyarekar, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
> On Oct 27, 2023, at 3:21 AM, Martin Uecker <uecker@tugraz.at> wrote:
>
> Am Donnerstag, dem 26.10.2023 um 19:57 +0000 schrieb Qing Zhao:
>> I guess that what Kees wanted, ""fill the array without knowing the actual final size" code pattern”, as following:
>>
>>>> struct foo *f;
>>>> char *p;
>>>> int i;
>>>>
>>>> f = alloc(maximum_possible);
>>>> f->count = 0;
>>>> p = f->buf;
>>>>
>>>> for (i; data_is_available() && i < maximum_possible; i++) {
>>>> f->count ++;
>>>> p[i] = next_data_item();
>>>> }
>>
>> actually is a dynamic array, or more accurately, Bounded-size dynamic array: ( but not a dynamic allocated array as we discussed so far)
>>
>> https://en.wikipedia.org/wiki/Dynamic_array
>>
>> This dynamic array, also is called growable array, or resizable array, whose size can
>> be changed during the lifetime.
>>
>> For VLA or FAM, I believe that they are both dynamic allocated array, i.e, even though the size is not know at the compilation time, but the size
>> will be fixed after the array is allocated.
>>
>> I am not sure whether C has support to such Dynamic array? Or whether it’s easy to provide dynamic array support in C?
>
> It is possible to support dynamic arrays in C even with
> good checking, but not safely using the pattern above
> where you derive a pointer which you later use independently.
>
> While we could track the connection to the original struct,
> the necessary synchronization between the counter and the
> access to the buffer is difficult. I do not see how this
> could be supported with reasonable effort and cost.
>
>
> But with this restriction in mind, we can do a lot in C.
> For example, see my experimental (!) container library
> which has vector type.
> https://github.com/uecker/noplate/blob/main/test.c
> You can get an array view for the vector (which then
> also can decay to a pointer), so it interoperates nicely
> with C but you can get good bounds checking.
>
>
> But once you derive a pointer and pass it on, it gets
> difficult. But if you want safety, you just have to
> to simply avoid this in code.
So, for the following modified code: (without the additional pointer “p”)
struct foo
{
size_t count;
char buf[] __attribute__((counted_by(count)));
};
struct foo *f;
int i;
f = alloc(maximum_possible);
f->count = 0;
for (i; data_is_available() && i < maximum_possible; i++) {
f->count ++;
f->buf[i] = next_data_item();
}
The support for dynamic array should be possible?
>
> What we could potentially do is add restrictions so
> that the access to buf always has to go via x->buf
> or you get at least a warning.
Are the following two restrictions to the user enough:
1. The access to buf should always go via x->buf,
no assignment to another independent pointer
and access buf through this new pointer.
2. User need to keep the synchronization between
the counter and the access to the buffer all the time.
Qing
>
> Martin
>
>
>
>
>>
>> Qing
>>
>>
>>> On Oct 26, 2023, at 12:45 PM, Martin Uecker <uecker@tugraz.at> wrote:
>>>
>>> Am Donnerstag, dem 26.10.2023 um 09:13 -0700 schrieb Kees Cook:
>>>> On Thu, Oct 26, 2023 at 10:15:10AM +0200, Martin Uecker wrote:
>>>>> but not this:
>>>>>
>>>
>>> x->count = 11;
>>>>> char *p = &x->buf;
>>>>> x->count = 1;
>>>>> p[10] = 1; // !
>>>>
>>>> This seems fine to me -- it's how I'd expect it to work: "10" is beyond
>>>> "1".
>>>
>>> Note that the store would be allowed.
>>>
>>>>
>>>>> (because the pointer is passed around the
>>>>> store to the counter)
>>>>>
>>>>> and also here the second store is then irrelevant
>>>>> for the access:
>>>>>
>>>>> x->count = 10;
>>>>> char* p = &x->buf;
>>>>> ...
>>>>> x->count = 1; // somewhere else
>>>>> ----
>>>>> p[9] = 1; // ok, because count matter when buf was accesssed.
>>>>
>>>> This is less great, but I can understand why it happens. "p" loses the
>>>> association with "x". It'd be nice if "p" had to way to retain that it
>>>> was just an alias for x->buf, so future p access would check count.
>>>
>>> The problem is not to discover that p is an alias to x->buf,
>>> but that it seems difficult to make sure that stores to
>>> x->count are not reordered relative to the final access to
>>> p[i] you want to check, so that you then get the right value.
>>>
>>>>
>>>> But this appears to be an existing limitation in other areas where an
>>>> assignment will cause the loss of object association. (I've run into
>>>> this before.) It's just more surprising in the above example because in
>>>> the past the loss of association would cause __bdos() to revert back to
>>>> "SIZE_MAX" results ("I don't know the size") rather than an "outdated"
>>>> size, which may get us into unexpected places...
>>>>
>>>>> IMHO this makes sense also from the user side and
>>>>> are the desirable semantics we discussed before.
>>>>>
>>>>> But can you take a look at this?
>>>>>
>>>>>
>>>>> This should simulate it fairly well:
>>>>> https://godbolt.org/z/xq89aM7Gr
>>>>>
>>>>> (the call to the noinline function would go away,
>>>>> but not necessarily its impact on optimization)
>>>>
>>>> Yeah, this example should be a very rare situation: a leaf function is
>>>> changing the characteristics of the struct but returning a buffer within
>>>> it to the caller. The more likely glitch would be from:
>>>>
>>>> int main()
>>>> {
>>>> struct foo *f = foo_alloc(7);
>>>> char *p = FAM_ACCESS(f, size, buf);
>>>>
>>>> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
>>>> test1(f); // or just "f->count = 10;" no function call needed
>>>> printf("%ld\n", __builtin_dynamic_object_size(p, 0));
>>>>
>>>> return 0;
>>>> }
>>>>
>>>> which reports:
>>>> 7
>>>> 7
>>>>
>>>> instead of:
>>>> 7
>>>> 10
>>>>
>>>> This kind of "get an alias" situation is pretty common in the kernel
>>>> as a way to have a convenient "handle" to the array. In the case of a
>>>> "fill the array without knowing the actual final size" code pattern,
>>>> things would immediately break:
>>>>
>>>> struct foo *f;
>>>> char *p;
>>>> int i;
>>>>
>>>> f = alloc(maximum_possible);
>>>> f->count = 0;
>>>> p = f->buf;
>>>>
>>>> for (i; data_is_available() && i < maximum_possible; i++) {
>>>> f->count ++;
>>>> p[i] = next_data_item();
>>>> }
>>>>
>>>> Now perhaps the problem here is that "count" cannot be used for a count
>>>> of "logically valid members in the array" but must always be a count of
>>>> "allocated member space in the array", which I guess is tolerable, but
>>>> isn't ideal -- I'd like to catch logic bugs in addition to allocation
>>>> bugs, but the latter is certainly much more important to catch.
>>>
>>> Maybe we could have a warning when f->buf is not directly
>>> accessed.
>>>
>>> Martin
>>>
>>>>
>>>
>>
>
^ permalink raw reply [flat|nested] 116+ messages in thread
* Re: HELP: Will the reordering happen? Re: [V3][PATCH 0/3] New attribute "counted_by" to annotate bounds for C99 FAM(PR108896)
2023-10-27 14:32 ` Qing Zhao
@ 2023-10-27 14:53 ` Martin Uecker
2023-10-27 15:10 ` Qing Zhao
0 siblings, 1 reply; 116+ messages in thread
From: Martin Uecker @ 2023-10-27 14:53 UTC (permalink / raw)
To: Qing Zhao
Cc: Kees Cook, Siddhesh Poyarekar, Richard Biener, Joseph Myers,
Jakub Jelinek, gcc Patches, isanbard
Am Freitag, dem 27.10.2023 um 14:32 +0000 schrieb Qing Zhao:
>
> > On Oct 27, 2023, at 3:21 AM, Martin Uecker <uecker@tugraz.at> wrote:
> >
> > Am Donnerstag, dem 26.10.2023 um 19:57 +0000 schrieb Qing Zhao:
> > > I guess that what Kees wanted, ""fill the array without knowing the actual final size" code pattern”, as following:
> > >
> > > > > struct foo *f;
> > > > > char *p;
> > > > > int i;
> > > > >
> > > > > f = alloc(maximum_possible);
> > > > > f->count = 0;
> > > > > p = f->buf;
> > > > >
> > > > > for (i; data_is_available() && i < maximum_possible; i++) {
> > > > > f->count ++;
> > > > > p[i] = next_data_item();
> > > > > }
> > >
> > > actually is a dynamic array, or more accurately, Bounded-size dynamic array: ( but not a dynamic allocated array as we discussed so far)
> > >
> > > https://en.wikipedia.org/wiki/Dynamic_array
> > >
> > > This dynamic array, also is called growable array, or resizable array, whose size can
> > > be changed during the lifetime.
> > >
> > > For VLA or FAM, I believe that they are both dynamic allocated array, i.e, even though the size is not know at the compilation time, but the size
> > > will be fixed after the array is allocated.
> > >
> > > I am not sure whether C has support to such Dynamic array? Or whether it’s easy to provide dynamic array support in C?
> >
> > It is possible to support dynamic arrays in C even with
> > good checking, but not safely using the pattern above
> > where you derive a pointer which you later use independently.
> >
> > While we could track the connection to the original struct,
> > the necessary synchronization between the counter and the
> > access to the buffer is difficult. I do not see how this
> > could be supported with reasonable effort and cost.
> >
> >
> > But with this restriction in mind, we can do a lot in C.
> > For example, see my experimental (!) container library
> > which has vector type.
> > https://github.com/uecker/noplate/blob/main/test.c
> > You can get an array view for the vector (which then
> > also can decay to a pointer), so it interoperates nicely
> > with C but you can get good bounds checking.
> >
> >
> > But once you derive a pointer and pass it on, it gets
> > difficult. But if you want safety, you just have to
> > to simply avoid this in code.
>
> So, for the following modified code: (without the additional pointer “p”)
>
> struct foo
> {
> size_t count;
> char buf[] __attribute__((counted_by(count)));
> };
>
> struct foo *f;
> int i;
>
> f = alloc(maximum_possible);
> f->count = 0;
>
> for (i; data_is_available() && i < maximum_possible; i++) {
> f->count ++;
> f->buf[i] = next_data_item();
> }
>
> The support for dynamic array should be possible?
With the design we discussed this should work because
__builtin_with_access (or whatever) it reads:
f = alloc(maximum_possible);
f->count = 0;
for (i; data_is_available() && i < maximum_possible; i++) {
f->count ++;
__builtin_with_access(f->buf, f->count)[i] = next_data_item();
}
>
>
> >
> > What we could potentially do is add restrictions so
> > that the access to buf always has to go via x->buf
> > or you get at least a warning.
>
> Are the following two restrictions to the user enough:
>
> 1. The