Pointer comparison
In CHERI C/C++, pointer comparison considers only the
integer address part of a capability.
This means that differences in tag validity, bounds, permissions, and so on,
will not be considered when by C operators such as ==
, <
, and <=
.
On the whole, this leads to intuitive behavior in systems software, where,
for example, malloc
adjusts bounds on a pointer before returning it to
a caller, and then expects an address-wise comparison to succeed when the
pointer is later returned via a call to free
.
However, this behavior could also lead to potentially confusing results; for
example:
-
If a tag on a pointer is lost due to non-provenance-preserving
memcpy
(e.g., afor
loop copying a sequence of bytes), the source and destination pointers will compare as equal even though the destination will not be dereferenceable. -
If a
realloc
implementation returns a pointer to the same address, but with different bounds, a caller check to see if the passed and returned pointers are equal will returntrue
even though an access might be permitted via one pointer but not the other.
However, practical experience has suggested that the current semantics produce fewer subtle bugs, and require fewer changes, than having comparison operators take the tag or other metadata into account.1
The CHERI Clang compiler supports an experimental flag -cheri-comparison=exact
that causes capability equality comparisons to also include capability metadata and the tag bit.