-
Notifications
You must be signed in to change notification settings - Fork 109
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Enable fat pointer comparison for slices and add check for vtable ones #1195
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Neat! Thanks Celina!
BinOp::Gt => self.codegen_operand(e1).gt(self.codegen_operand(e2)), | ||
_ => unreachable!(), | ||
} | ||
let left = self.codegen_operand(e1); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice refactoring!
// Codegen an assertion failure since vtable comparison is not stable. | ||
let ret_type = Type::Bool; | ||
let body = vec![ | ||
self.codegen_assert_false( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is it not possible to use codegen_unimplemented
instead?
EDIT: I now see that this is due to the property class. Perhaps this needs refactoring to add a property-class parameter in a follow-up PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think this should use codegen_unimplemented()
though. This is by design and it's meant to alert users of behaviour instability.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we should only use codegen_unimplemented
for things that Kani can't yet handle.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps we can put the code under codegen_unimplemented
in a function with a more generic name (e.g. codegen_assert_assume_false
), and call that function from codegen_unimplemented
and codegen_unstable
?
EDIT: Not necessarily in this PR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My bad, I think we already have that. codegen_fatal_error()
should do it. It uses Abort instead of assume.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It actually takes Span instead of Location. So I'll leave this change for another time. Sorry!
// It is different if any is different. | ||
BinOp::Ne => data_cmp.or(metadata_cmp), | ||
// If data is different, only compare data. | ||
// If data is equal, apply operator to metadata. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Any references for this logic?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure. Let me find the PR that implemented this on the rust compiler side.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, we need a reference here.
_ => unreachable!(), | ||
} | ||
let left = self.codegen_operand(e1); | ||
let right = self.codegen_operand(e2); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nit: left_op
and right_op
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
sure
self.codegen_assert_false( | ||
PropertyClass::KaniCheck, | ||
format!("Reached unstable vtable comparison '{:?}'", op).as_str(), | ||
Location::None, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is there a way to get the locations for these statements? If so, I suggest adding them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can try to refactor things a bit to pass down the location.
// It is different if any is different. | ||
BinOp::Ne => data_cmp.or(metadata_cmp), | ||
// If data is different, only compare data. | ||
// If data is equal, apply operator to metadata. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, we need a reference here.
1. This only works for pointers with the same provenance. 2. We should model vtable comparison a bit differently. Follow up commit
- Rename variables. - Add location to the assertion and other statements. - Added a link to the compiler PR that implemented fat pointer comparison. I did not change the assert / assume logic to a different function yet.
Co-authored-by: Zyad Hassan <[email protected]>
Description of changes:
Currently, we don't support fat pointer comparison. This issue adds support for slice fat pointer comparison and a proper check that will fail the verification if its trying to compare vtable fat pointers.
Resolved issues:
Resolves #327
Call-outs:
I documented in the original issue #327 why I don't think we should compare
vtable
fat pointers. In a nutshell, this operation is not UB but it is unstable. I.e.: The behavior may depend on the compilation / linking process.I kept a similar behavior to thin pointers when comparing the data pointer portion of the fat pointer. Comparing pointers that have different precedence will also fail.
We could add optional support to the operations mentioned above using uninterpreted functions, but I will defer that to when we see demand for it.
Testing:
How is this change tested? New tests
Is this a refactor change? No
Checklist
By submitting this pull request, I confirm that my contribution is made under the terms of the Apache 2.0 and MIT licenses.