Subobject bounds
CHERI C/C++ also supports automatically restricting the
bounds when a pointer is taken to a subobject — for example, an array
embedded within another structure that itself has been heap allocated.
This will prevent an overflow on that array from affecting the remainder of
the structure, improving spatial safety.
Subobject bounds are not enabled by default as they may require additional source code changes
for compatibility, but can be enabled using the -Xclang -cheri-bounds=subobject-safe
compiler flag.
One example of C code that requires changes for subobject bounds is the containerof
pattern, in which pointer arithmetic on a pointer to a subobject is used to
recover a pointer to the container object — for example, as seen in the
widely used BSD queue.h
linked-list macros or the generic C
hash-table implementation, uthash.h
.
In these cases, an opt-out annotation can be applied to a given type, field or variable that instructs the compiler to not tighten bounds when creating pointers to subobjects. We currently define three opt-out annotations that can be used to allow existing code to disable use of subobject bounds:
Completely disable subobject bounds: It is possible to annotate a typedef, record member, or variable declaration with:
__attribute__((cheri_no_subobject_bounds))
to indicate that the compiler should not tighten bounds when taking the address or a C++ reference. In C++11/C20 mode this can also be spelled as [[cheri::no_subobject_bounds]]
.
struct str {
/*
* Nul-terminated string array -- pointers taken to this subobject will
* use the array's bounds, not those of the container structure.
*/
char str_array[128];
/*
* Linked-list entry element -- because of the additional attribute,
* pointers taken to this subobject will use the container structure's
* bounds, not those of the specific field.
*/
struct list_entry str_le __attribute__((cheri_no_subobject_bounds));
} str_instance;
void
fn(void)
{
/* Struct pointer gets bounds of str_instance. */
struct str *strp = &str_instance;
/* Character pointer gets bounds of the subobject, not str_instance. */
char *c = str_instance.str_array;
/* Struct pointer gets bounds of str_instance, not the subobject. */
struct list_entry *lep = &str_instance.str_le;
}
Disable subobject bounds in specific expressions: It is also possible to opt out of bounds-tightening on a per-expression granularity by casting to an annotated type:
char *foo(struct str *strp) {
return (&((__attribute__((cheri_no_subobject_bounds))struct str *)
strp)->str_array);
}
Use remaining allocation size: In certain cases, the size of the subobject is not known, but we still know that data before the field member will not be accessed (e.g., variable size array members inside structs). Pre-C99 code will declare such members as fixed-size arrays, which will cause a hardware exception if the allocation does not grant access to that many bytes. 1 To use the remaining allocation size instead of completely disabling bounds (and thus protecting against buffer underflows) the annotation:
__attribute__((cheri_subobject_bounds_use_remaining_size))
can be used. When targeting C++11/C20:
[[cheri::subobject_bounds_use_remaining_size]]
is also supported.
Examples of this pattern include FreeBSD's struct dirent
, which uses
char d_name[255]
for an array that is actually of variable size, with
the containing allocation (e.g., of the heap) being sized to allow additional
space for array entries regardless of size in the type definition.
For example:
struct message {
int m_type;
/*
* Variable-length character array -- because of the additional
* attribute, pointers taken to this subobject will have a lower bound
* at the first address of the array, but retain an upper bound of the
* allocation containing the array, rather than 252 bytes higher.
*/
char m_data[252]
__attribute__((cheri_subobject_bounds_use_remaining_size));
};
The use of subobject bounds imposes additional compatibility constraints on existing C and C++ code. While we have not encountered many issues related to subobject bounds in existing code, it does slightly increase the porting effort.
If flexible arrays members are declared using the C99 syntax with empty square brackets, the compiler will automatically use the remaining allocation size.